tx · 9wWnRUe4hJ9Q2Q4HuC4Kg7efGpPsfjQuZYk6S6ES2RZ3 3N1TdY92brvvrQyVdkpHNK2aRbAsx9dYUKc: -0.01000000 Waves 2022.05.23 09:43 [2063987] smart account 3N1TdY92brvvrQyVdkpHNK2aRbAsx9dYUKc > SELF 0.00000000 Waves
{ "type": 13, "id": "9wWnRUe4hJ9Q2Q4HuC4Kg7efGpPsfjQuZYk6S6ES2RZ3", "fee": 1000000, "feeAssetId": null, "timestamp": 1653288221828, "version": 2, "chainId": 84, "sender": "3N1TdY92brvvrQyVdkpHNK2aRbAsx9dYUKc", "senderPublicKey": "3N3TSucEwKCeSJy32W5m6LB2taq7FG7aQ7XxiXvJGdCk", "proofs": [ "4XFyskgVQAJAWtiAp91BUw2uBLppgMn2M9eQsQrnJVJZFZorWQpAg45mcei59Qepecs3qMLCp3YY4V7BnuJfrMok" ], "script": "base64:AAIFAAAAAAAAACwIAhIGCgQCAgICEggKBgICAQICAhIHCgUCAgECAhIDCgECEgMKAQISAwoBAgAAABUAAAAAEGtleUxvY2tSZWNpcGllbnQCAAAAA19scgAAAAANa2V5TG9ja0Ftb3VudAIAAAADX2xhAAAAABJrZXlMb2NrRGVzdGluYXRpb24CAAAAA19sZAAAAAASa2V5TG9ja0Fzc2V0U291cmNlAgAAAARfbGFzAAAAAAprZXlWZXJzaW9uAgAAAAJfdgAAAAAJa2V5QnJpZGdlAgAAAAJfYgAAAAAJa2V5T3JhY2xlAgAAAAJfbwAAAAAIa2V5QWRtaW4CAAAAAl9hAAAAAAlrZXlVbmxvY2sCAAAAAl91AAAAAA9lcnJVbmF1dGhvcml6ZWQCAAAADHVuYXV0aG9yaXplZAAAAAAQZXJyVW5pbml0aWFsaXplZAIAAAANdW5pbml0aWFsaXplZAAAAAATZXJySW52YWxpZFNpZ25hdHVyZQIAAAARaW52YWxpZCBzaWduYXR1cmUAAAAAEWVyckFscmVhZHlDbGFpbWVkAgAAAAdjbGFpbWVkAAAAABBlcnJBbHJlYWR5TG9ja2VkAgAAAAZsb2NrZWQAAAAAEGVyckludmFsaWRMb2NrSWQCAAAADmludmFsaWQgbG9ja0lkAAAAAApjaGFpbldhdmVzAgAAAAhWMEZXUlE9PQEAAAAMYXNzZXJ0Q2FsbGVyAAAAAgAAAAFpAAAAA2tleQQAAAAHJG1hdGNoMAkABCEAAAABBQAAAANrZXkDCQAAAQAAAAIFAAAAByRtYXRjaDACAAAACkJ5dGVWZWN0b3IEAAAACWF1dGhvcml0eQUAAAAHJG1hdGNoMAMJAQAAAAIhPQAAAAIFAAAACWF1dGhvcml0eQgIBQAAAAFpAAAABmNhbGxlcgAAAAVieXRlcwkAAAIAAAABBQAAAA9lcnJVbmF1dGhvcml6ZWQFAAAABHVuaXQJAAACAAAAAQUAAAAQZXJyVW5pbml0aWFsaXplZAEAAAAUYXNzZXJ0Q2FsbGVySXNCcmlkZ2UAAAABAAAAAWkJAQAAAAxhc3NlcnRDYWxsZXIAAAACBQAAAAFpBQAAAAlrZXlCcmlkZ2UBAAAAE2Fzc2VydENhbGxlcklzQWRtaW4AAAABAAAAAWkJAQAAAAxhc3Nl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BgUAAAADbmlsBgkAAAIAAAABAgAAACRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAACAAAAAQIAAAAkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAAAAAWkBAAAACmNyZWF0ZUxvY2sAAAAFAAAABmxvY2tJZAAAAAlyZWNpcGllbnQAAAAGYW1vdW50AAAAD2xvY2tEZXN0aW5hdGlvbgAAABVhc3NldFNvdXJjZUFuZEFkZHJlc3MEAAAACWxvY2tJZFN0cgkAAloAAAABBQAAAAZsb2NrSWQEAAAAEGxvY2tSZWNpcGllbnRLZXkJAAEsAAAAAgUAAAAJbG9ja0lkU3RyBQAAABBrZXlMb2NrUmVjaXBpZW50BAAAAAtjaGVja0NhbGxlcgkBAAAAFGFzc2VydENhbGxlcklzQnJpZGdlAAAAAQUAAAABaQMJAAAAAAAAAgUAAAALY2hlY2tDYWxsZXIFAAAAC2NoZWNrQ2FsbGVyBAAAAAtjaGVja0xvY2tJZAkBAAAAEWFzc2VydFZhbGlkTG9ja0lkAAAAAQUAAAAGbG9ja0lkAwkAAAAAAAACBQAAAAtjaGVja0xvY2tJZAUAAAALY2hlY2tMb2NrSWQEAAAACWNoZWNrTG9jawkBAAAAE2Fzc2VydExvY2tOb3RFeGlzdHMAAAABBQAAABBsb2NrUmVjaXBpZW50S2V5AwkAAAAAAAACBQAAAAljaGVja0xvY2sFAAAACWNoZWNrTG9jawkABEwAAAACCQEAAAALQmluYXJ5RW50cnkAAAACCQABLAAAAAIFAAAACWxvY2tJZFN0cgUAAAAQa2V5TG9ja1JlY2lwaWVudAUAAAAJcmVjaXBpZW50CQAETAAAAAIJAQAAAAxJbnRlZ2VyRW50cnkAAAACCQABLAAAAAIFAAAACWxvY2tJZFN0cgUAAAANa2V5TG9ja0Ftb3VudAUAAAAGYW1vdW50CQAETAAAAAIJAQAAAAtCaW5hcnlFbnRyeQAAAAIJAAEsAAAAAgUAAAAJbG9ja0lkU3RyBQAAABJrZXlMb2NrRGVzdGluYXRpb24FAAAAD2xvY2tEZXN0aW5hdGlvbgkABEwAAAACCQEAAAALQmluYXJ5RW50cnkAAAACCQABLAAAAAIFAAAACWxvY2tJZFN0cgUAAAASa2V5TG9ja0Fzc2V0U291cmNlBQAAABVhc3NldFNvdXJjZUFuZEFkZHJlc3MFAAAAA25pbAkAAAIAAAABAgAAACRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4JAAACAAAAAQIAAAAkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuCQAAAgAAAAECAAAAJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgAAAAFpAQAAAAhzZXRBZG1pbgAAAAEAAAAIbmV3QWRtaW4EAAAAC2NoZWNrQ2FsbGVyCQEAAAATYXNzZXJ0Q2FsbGVySXNBZG1pbgAAAAEFAAAAAWkDCQAAAAAAAAIFAAAAC2NoZWNrQ2FsbGVyBQAAAAtjaGVja0NhbGxlcgkABEwAAAACCQEAAAALQmluYXJ5RW50cnkAAAACBQAAAAhrZXlBZG1pbgUAAAAIbmV3QWRtaW4FAAAAA25pbAkAAAIAAAABAgAAACRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4AAAABaQEAAAAJc2V0T3JhY2xlAAAAAQAAAAluZXdPcmFjbGUEAAAAC2NoZWNrQ2FsbGVyCQEAAAATYXNzZXJ0Q2FsbGVySXNBZG1pbgAAAAEFAAAAAWkDCQAAAAAAAAIFAAAAC2NoZWNrQ2FsbGVyBQAAAAtjaGVja0NhbGxlcgkABEwAAAACCQEAAAALQmluYXJ5RW50cnkAAAACBQAAAAlrZXlPcmFjbGUFAAAACW5ld09yYWNsZQUAAAADbmlsCQAAAgAAAAECAAAAJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgAAAAFpAQAAAAlzZXRCcmlkZ2UAAAABAAAACW5ld0JyaWRnZQQAAAALY2hlY2tDYWxsZXIJAQAAABNhc3NlcnRDYWxsZXJJc0FkbWluAAAAAQUAAAABaQMJAAAAAAAAAgUAAAALY2hlY2tDYWxsZXIFAAAAC2NoZWNrQ2FsbGVyCQAETAAAAAIJAQAAAAtCaW5hcnlFbnRyeQAAAAIFAAAACWtleUJyaWRnZQUAAAAJbmV3QnJpZGdlBQAAAANuaWwJAAACAAAAAQIAAAAkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuAAAAAJlbLbE=", "height": 2063987, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: 2xBXLq3BhTVoBWxSLd7zx4FvsjeQumZ4kgJFyTa5824w Next: none Diff:
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 5 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | - | let | |
4 | + | let keyLockRecipient = "_lr" | |
5 | 5 | ||
6 | - | let | |
6 | + | let keyLockAmount = "_la" | |
7 | 7 | ||
8 | - | let | |
8 | + | let keyLockDestination = "_ld" | |
9 | 9 | ||
10 | - | let | |
10 | + | let keyLockAssetSource = "_las" | |
11 | 11 | ||
12 | - | let | |
12 | + | let keyVersion = "_v" | |
13 | 13 | ||
14 | - | let | |
14 | + | let keyBridge = "_b" | |
15 | 15 | ||
16 | - | let | |
16 | + | let keyOracle = "_o" | |
17 | 17 | ||
18 | - | let | |
18 | + | let keyAdmin = "_a" | |
19 | 19 | ||
20 | - | let keyUnlockSigner = "_us" | |
21 | - | ||
22 | - | let keyIsActive = "_ia" | |
23 | - | ||
24 | - | let keyManager = "_m" | |
25 | - | ||
26 | - | let BRIDGE_MANAGER = "BRIDGE_MANAGER" | |
27 | - | ||
28 | - | let ASSET_MANAGER = "ASSET_MANAGER" | |
29 | - | ||
30 | - | let STOP_MANAGER = "STOP_MANAGER" | |
31 | - | ||
32 | - | let baseAssetSourceAndAddress = base58'fRpRFUxiLXbzAaAT3sfi1oTFa8n4X8b9FUaQHyT5MkdXYpGZV' | |
33 | - | ||
34 | - | let baseAssetId = base58'6scFjhFGDfpmYySMKQ9vDbZuH8aMRWsUQJAHXzm1FsJo' | |
35 | - | ||
36 | - | let chainWaves = base58'3EMsPJ' | |
37 | - | ||
38 | - | let BP = 10000 | |
39 | - | ||
40 | - | let systemPrecision = 9 | |
20 | + | let keyUnlock = "_u" | |
41 | 21 | ||
42 | 22 | let errUnauthorized = "unauthorized" | |
43 | 23 | ||
44 | 24 | let errUninitialized = "uninitialized" | |
45 | 25 | ||
46 | - | let | |
26 | + | let errInvalidSignature = "invalid signature" | |
47 | 27 | ||
48 | - | let | |
28 | + | let errAlreadyClaimed = "claimed" | |
49 | 29 | ||
50 | - | let | |
30 | + | let errAlreadyLocked = "locked" | |
51 | 31 | ||
52 | - | let | |
32 | + | let errInvalidLockId = "invalid lockId" | |
53 | 33 | ||
54 | - | let | |
34 | + | let chainWaves = "V0FWRQ==" | |
55 | 35 | ||
56 | - | let errAssetNotFound = "asset not found" | |
57 | - | ||
58 | - | let errNotEnoughBalance = "not enough balance" | |
59 | - | ||
60 | - | let errBigPowValue = "pow value too big" | |
61 | - | ||
62 | - | let errBridgeDisabled = "bridge is disabled" | |
63 | - | ||
64 | - | let errAssetDisabled = "asset is disabled" | |
65 | - | ||
66 | - | let errWrongDestinationChain = "wrong destination chain" | |
67 | - | ||
68 | - | let typeBase = 0 | |
69 | - | ||
70 | - | let typeNative = 1 | |
71 | - | ||
72 | - | let typeWrapped = 2 | |
73 | - | ||
74 | - | func assertCallerIsManager (i,managerType) = match getBinary((managerType + keyManager)) { | |
36 | + | func assertCaller (i,key) = match getBinary(key) { | |
75 | 37 | case authority: ByteVector => | |
76 | 38 | if ((authority != i.caller.bytes)) | |
77 | 39 | then throw(errUnauthorized) | |
81 | 43 | } | |
82 | 44 | ||
83 | 45 | ||
84 | - | func assertBridgeIsActive () = if (!(valueOrElse(getBoolean(keyIsActive), false))) | |
85 | - | then throw(errBridgeDisabled) | |
86 | - | else unit | |
46 | + | func assertCallerIsBridge (i) = assertCaller(i, keyBridge) | |
87 | 47 | ||
88 | 48 | ||
89 | - | func assertAssetIsActive (assetIdStr) = match getBoolean((assetIdStr + keyAssetIsActive)) { | |
90 | - | case isActive: Boolean => | |
91 | - | if (isActive) | |
92 | - | then unit | |
93 | - | else throw(errAssetDisabled) | |
49 | + | func assertCallerIsAdmin (i) = assertCaller(i, keyAdmin) | |
50 | + | ||
51 | + | ||
52 | + | func assertValidLockId (lockId) = { | |
53 | + | let version = valueOrErrorMessage(getBinary(keyVersion), errUninitialized) | |
54 | + | if (if ((size(lockId) != 16)) | |
55 | + | then true | |
56 | + | else (take(lockId, 1) != version)) | |
57 | + | then throw(errInvalidLockId) | |
58 | + | else unit | |
59 | + | } | |
60 | + | ||
61 | + | ||
62 | + | func assertLockNotExists (lockRecipientKey) = match getBinary(lockRecipientKey) { | |
63 | + | case t: ByteVector => | |
64 | + | throw(errAlreadyLocked) | |
94 | 65 | case _ => | |
95 | - | | |
66 | + | unit | |
96 | 67 | } | |
97 | 68 | ||
98 | 69 | ||
99 | - | func getFee (amount,assetIdStr) = { | |
100 | - | let baseFeeRateBP = valueOrErrorMessage(getInteger(keyBaseFeeRateBP), errUninitialized) | |
101 | - | let minFee = valueOrErrorMessage(getInteger((assetIdStr + keyAssetMinFee)), errAssetNotFound) | |
102 | - | let fee = ((amount * baseFeeRateBP) / BP) | |
103 | - | if ((minFee > fee)) | |
104 | - | then minFee | |
105 | - | else fee | |
106 | - | } | |
107 | - | ||
108 | - | ||
109 | - | func pow10 (value,power) = { | |
110 | - | let absPow = if ((0 > power)) | |
111 | - | then -(power) | |
112 | - | else power | |
113 | - | let powerValue = if ((absPow == 0)) | |
114 | - | then 1 | |
115 | - | else if ((absPow == 1)) | |
116 | - | then 10 | |
117 | - | else if ((absPow == 2)) | |
118 | - | then 100 | |
119 | - | else if ((absPow == 3)) | |
120 | - | then 1000 | |
121 | - | else if ((absPow == 4)) | |
122 | - | then 10000 | |
123 | - | else if ((absPow == 5)) | |
124 | - | then 100000 | |
125 | - | else if ((absPow == 6)) | |
126 | - | then 1000000 | |
127 | - | else if ((absPow == 7)) | |
128 | - | then 10000000 | |
129 | - | else if ((absPow == 8)) | |
130 | - | then 100000000 | |
131 | - | else if ((absPow == 9)) | |
132 | - | then 1000000000 | |
133 | - | else throw(errBigPowValue) | |
134 | - | if ((0 > power)) | |
135 | - | then (value / powerValue) | |
136 | - | else (value * powerValue) | |
137 | - | } | |
138 | - | ||
139 | - | ||
140 | - | func toSystemPrecision (amount,precision) = pow10(amount, (systemPrecision - precision)) | |
141 | - | ||
142 | - | ||
143 | - | func fromSystemPrecision (amount,precision) = pow10(amount, (precision - systemPrecision)) | |
144 | - | ||
145 | - | ||
146 | 70 | @Callable(i) | |
147 | - | func init (admin, | |
148 | - | then throw( | |
149 | - | else [BinaryEntry( | |
71 | + | func init (admin,version,bridge,oracle) = if (isDefined(getBinary(keyAdmin))) | |
72 | + | then throw(errUnauthorized) | |
73 | + | else [BinaryEntry(keyVersion, version), BinaryEntry(keyBridge, bridge), BinaryEntry(keyOracle, oracle), BinaryEntry(keyAdmin, admin)] | |
150 | 74 | ||
151 | 75 | ||
152 | 76 | ||
153 | 77 | @Callable(i) | |
154 | - | func | |
155 | - | let | |
156 | - | if (( | |
78 | + | func createUnlock (lockId,recipient,amount,lockSource,tokenSourceAndAddress,signature) = { | |
79 | + | let checkLockId = assertValidLockId(lockId) | |
80 | + | if ((checkLockId == checkLockId)) | |
157 | 81 | then { | |
158 | - | let $t041684568 = if ((assetId == baseAssetId)) | |
159 | - | then if ((assetSourceAndAddress != baseAssetSourceAndAddress)) | |
160 | - | then throw(errInvalidValues) | |
161 | - | else $Tuple2(typeBase, 8) | |
162 | - | else { | |
163 | - | let asset = valueOrErrorMessage(assetInfo(assetId), errAssetNotFound) | |
164 | - | let type = if ((asset.issuer == this)) | |
165 | - | then typeWrapped | |
166 | - | else typeNative | |
167 | - | $Tuple2(type, asset.decimals) | |
168 | - | } | |
169 | - | let type = $t041684568._1 | |
170 | - | let precision = $t041684568._2 | |
171 | - | let assetIdStr = toBase64String(assetId) | |
172 | - | let keySourceAddress = (toBase64String(assetSourceAndAddress) + keyAssetAddress) | |
173 | - | let keyNativeAddress = (assetIdStr + keyAssetAddress) | |
174 | - | if (if (isDefined(getBinary(keySourceAddress))) | |
175 | - | then true | |
176 | - | else isDefined(getBinary(keyNativeAddress))) | |
177 | - | then throw(errAlreadyExists) | |
178 | - | else [BinaryEntry(keySourceAddress, assetId), BinaryEntry(keyNativeAddress, assetSourceAndAddress), IntegerEntry((assetIdStr + keyAssetType), type), IntegerEntry((assetIdStr + keyAssetPrecision), precision), IntegerEntry((assetIdStr + keyAssetMinFee), minFee), BooleanEntry((assetIdStr + keyAssetIsActive), true)] | |
179 | - | } | |
180 | - | else throw("Strict value is not equal to itself.") | |
181 | - | } | |
182 | - | ||
183 | - | ||
184 | - | ||
185 | - | @Callable(i) | |
186 | - | func issue (name,description,precision) = { | |
187 | - | let callerCheck = assertCallerIsManager(i, ASSET_MANAGER) | |
188 | - | if ((callerCheck == callerCheck)) | |
189 | - | then [Issue(name, description, 0, precision, true)] | |
190 | - | else throw("Strict value is not equal to itself.") | |
191 | - | } | |
192 | - | ||
193 | - | ||
194 | - | ||
195 | - | @Callable(i) | |
196 | - | func removeAsset (assetSourceAndAddress,newAuthority) = { | |
197 | - | let callerCheck = assertCallerIsManager(i, ASSET_MANAGER) | |
198 | - | if ((callerCheck == callerCheck)) | |
199 | - | then { | |
200 | - | let keySourceAddress = (toBase64String(assetSourceAndAddress) + keyAssetAddress) | |
201 | - | let assetId = valueOrErrorMessage(getBinary(keySourceAddress), errNotExists) | |
202 | - | let assetIdStr = toBase64String(assetId) | |
203 | - | let type = valueOrErrorMessage(getInteger((assetIdStr + keyAssetType)), errNotExists) | |
204 | - | let actions = if ((type == typeBase)) | |
82 | + | let checkCaller = assertCallerIsBridge(i) | |
83 | + | if ((checkCaller == checkCaller)) | |
205 | 84 | then { | |
206 | - | let balance = wavesBalance(this) | |
207 | - | if ((balance.available > 0)) | |
208 | - | then [ScriptTransfer(Address(newAuthority), balance.available, unit)] | |
209 | - | else nil | |
210 | - | } | |
211 | - | else if ((type == typeNative)) | |
212 | - | then { | |
213 | - | let balance = assetBalance(this, assetId) | |
214 | - | if ((balance > 0)) | |
215 | - | then [ScriptTransfer(Address(newAuthority), balance, assetId)] | |
216 | - | else nil | |
217 | - | } | |
218 | - | else if ((type == typeWrapped)) | |
219 | - | then nil | |
220 | - | else throw(errInvalidValues) | |
221 | - | (actions ++ [DeleteEntry(keySourceAddress), DeleteEntry((assetIdStr + keyAssetAddress)), DeleteEntry((assetIdStr + keyAssetType)), DeleteEntry((assetIdStr + keyAssetPrecision)), DeleteEntry((assetIdStr + keyAssetMinFee)), DeleteEntry((assetIdStr + keyAssetIsActive))]) | |
222 | - | } | |
223 | - | else throw("Strict value is not equal to itself.") | |
224 | - | } | |
225 | - | ||
226 | - | ||
227 | - | ||
228 | - | @Callable(i) | |
229 | - | func lock (lockId,recipient,destination) = { | |
230 | - | let birdgeCheck = assertBridgeIsActive() | |
231 | - | if ((birdgeCheck == birdgeCheck)) | |
232 | - | then if ((destination == chainWaves)) | |
233 | - | then throw(errWrongDestinationChain) | |
234 | - | else if ((size(i.payments) != 1)) | |
235 | - | then throw(errNotOnePayment) | |
236 | - | else { | |
237 | - | let assetId = valueOrElse(i.payments[0].assetId, baseAssetId) | |
238 | - | let assetIdStr = toBase64String(assetId) | |
239 | - | let assetCheck = assertAssetIsActive(assetIdStr) | |
240 | - | if ((assetCheck == assetCheck)) | |
241 | - | then { | |
242 | - | let amount = i.payments[0].amount | |
243 | - | let validatorAddress = valueOrErrorMessage(getBinary(keyValidator), errUninitialized) | |
244 | - | let fee = getFee(amount, assetIdStr) | |
245 | - | let amountWithoutFee = (amount - fee) | |
246 | - | if ((0 >= amountWithoutFee)) | |
247 | - | then throw(errNotEnoughBalance) | |
248 | - | else { | |
249 | - | let assetSourceAndAddress = valueOrErrorMessage(getBinary((assetIdStr + keyAssetAddress)), errAssetNotFound) | |
250 | - | let type = valueOrErrorMessage(getInteger((assetIdStr + keyAssetType)), errAssetNotFound) | |
251 | - | let precision = valueOrErrorMessage(getInteger((assetIdStr + keyAssetPrecision)), errAssetNotFound) | |
252 | - | let feeCollector = valueOrErrorMessage(getBinary(keyFeeCollector), errUninitialized) | |
253 | - | let createLock = invoke(Address(validatorAddress), "createLock", [lockId, recipient, toSystemPrecision(amountWithoutFee, precision), destination, assetSourceAndAddress], nil) | |
254 | - | if ((createLock == createLock)) | |
255 | - | then if ((type == typeBase)) | |
256 | - | then [ScriptTransfer(Address(feeCollector), fee, unit)] | |
257 | - | else if ((type == typeNative)) | |
258 | - | then [ScriptTransfer(Address(feeCollector), fee, assetId)] | |
259 | - | else if ((type == typeWrapped)) | |
260 | - | then [Burn(assetId, amountWithoutFee), ScriptTransfer(Address(feeCollector), fee, assetId)] | |
261 | - | else throw(errInvalidValues) | |
262 | - | else throw("Strict value is not equal to itself.") | |
263 | - | } | |
264 | - | } | |
265 | - | else throw("Strict value is not equal to itself.") | |
266 | - | } | |
267 | - | else throw("Strict value is not equal to itself.") | |
268 | - | } | |
269 | - | ||
270 | - | ||
271 | - | ||
272 | - | @Callable(i) | |
273 | - | func unlock (lockId,recipient,amount,lockSource,assetSourceAndAddress,signature) = { | |
274 | - | let birdgeCheck = assertBridgeIsActive() | |
275 | - | if ((birdgeCheck == birdgeCheck)) | |
276 | - | then { | |
277 | - | let assetId = valueOrErrorMessage(getBinary((toBase64String(assetSourceAndAddress) + keyAssetAddress)), errAssetNotFound) | |
278 | - | let assetIdStr = toBase64String(assetId) | |
279 | - | let type = valueOrErrorMessage(getInteger((assetIdStr + keyAssetType)), errAssetNotFound) | |
280 | - | let precision = valueOrErrorMessage(getInteger((assetIdStr + keyAssetPrecision)), errAssetNotFound) | |
281 | - | let validatorAddress = valueOrErrorMessage(getBinary(keyValidator), errUninitialized) | |
282 | - | let unlockSigner = valueOrErrorMessage(getBinary(keyUnlockSigner), errUninitialized) | |
283 | - | let feeCollector = valueOrErrorMessage(getBinary(keyFeeCollector), errUninitialized) | |
284 | - | let createUnlock = invoke(Address(validatorAddress), "createUnlock", [lockId, recipient, amount, lockSource, assetSourceAndAddress, signature], nil) | |
285 | - | if ((createUnlock == createUnlock)) | |
286 | - | then { | |
287 | - | let isValid = match createUnlock { | |
288 | - | case v: Boolean => | |
289 | - | v | |
290 | - | case _ => | |
291 | - | false | |
292 | - | } | |
293 | - | if (!(isValid)) | |
294 | - | then throw(errInvalidValues) | |
85 | + | let lockIdStr = toBase64String(lockId) | |
86 | + | let lockSourceStr = toBase64String(lockSource) | |
87 | + | let unlockKey = (((lockSourceStr + "_") + lockIdStr) + keyUnlock) | |
88 | + | if (isDefined(getBoolean(unlockKey))) | |
89 | + | then throw(errAlreadyClaimed) | |
295 | 90 | else { | |
296 | - | let fee = if ((unlockSigner == i.caller.bytes)) | |
297 | - | then valueOrErrorMessage(getInteger((assetIdStr + keyAssetMinFee)), errAssetNotFound) | |
298 | - | else 0 | |
299 | - | let amountToSend = fromSystemPrecision(amount, precision) | |
300 | - | let amountToSendWithoutFee = (amountToSend - fee) | |
301 | - | if ((0 >= amountToSendWithoutFee)) | |
302 | - | then throw(errNotEnoughBalance) | |
303 | - | else if ((type == typeBase)) | |
304 | - | then ([ScriptTransfer(Address(recipient), amountToSendWithoutFee, unit)] ++ (if ((fee > 0)) | |
305 | - | then [ScriptTransfer(Address(feeCollector), fee, unit)] | |
306 | - | else nil)) | |
307 | - | else if ((type == typeNative)) | |
308 | - | then ([ScriptTransfer(Address(recipient), amountToSendWithoutFee, assetId)] ++ (if ((fee > 0)) | |
309 | - | then [ScriptTransfer(Address(feeCollector), fee, assetId)] | |
310 | - | else nil)) | |
311 | - | else if ((type == typeWrapped)) | |
312 | - | then ([Reissue(assetId, amountToSend, true), ScriptTransfer(Address(recipient), amountToSendWithoutFee, assetId)] ++ (if ((fee > 0)) | |
313 | - | then [ScriptTransfer(Address(feeCollector), fee, assetId)] | |
314 | - | else nil)) | |
315 | - | else throw(errInvalidValues) | |
91 | + | let message = makeString([lockIdStr, toBase64String(recipient), toString(amount), lockSourceStr, toBase64String(tokenSourceAndAddress), chainWaves], "_") | |
92 | + | let hash = keccak256(toBytes(message)) | |
93 | + | let recoveredKey = ecrecover(hash, signature) | |
94 | + | if ((recoveredKey != getBinaryValue(keyOracle))) | |
95 | + | then throw(errInvalidSignature) | |
96 | + | else $Tuple2([BooleanEntry(unlockKey, true)], true) | |
316 | 97 | } | |
317 | 98 | } | |
318 | 99 | else throw("Strict value is not equal to itself.") | |
323 | 104 | ||
324 | 105 | ||
325 | 106 | @Callable(i) | |
326 | - | func setManager (managerType,manager) = { | |
327 | - | let callerCheck = assertCallerIsManager(i, BRIDGE_MANAGER) | |
328 | - | if ((callerCheck == callerCheck)) | |
329 | - | then [BinaryEntry((managerType + keyManager), manager)] | |
330 | - | else throw("Strict value is not equal to itself.") | |
331 | - | } | |
332 | - | ||
333 | - | ||
334 | - | ||
335 | - | @Callable(i) | |
336 | - | func setFeeCollector (feeCollector) = { | |
337 | - | let callerCheck = assertCallerIsManager(i, BRIDGE_MANAGER) | |
338 | - | if ((callerCheck == callerCheck)) | |
339 | - | then [BinaryEntry(keyFeeCollector, feeCollector)] | |
340 | - | else throw("Strict value is not equal to itself.") | |
341 | - | } | |
342 | - | ||
343 | - | ||
344 | - | ||
345 | - | @Callable(i) | |
346 | - | func setValidator (validator) = { | |
347 | - | let callerCheck = assertCallerIsManager(i, BRIDGE_MANAGER) | |
348 | - | if ((callerCheck == callerCheck)) | |
349 | - | then [BinaryEntry(keyValidator, validator)] | |
350 | - | else throw("Strict value is not equal to itself.") | |
351 | - | } | |
352 | - | ||
353 | - | ||
354 | - | ||
355 | - | @Callable(i) | |
356 | - | func startBridge () = { | |
357 | - | let callerCheck = assertCallerIsManager(i, BRIDGE_MANAGER) | |
358 | - | if ((callerCheck == callerCheck)) | |
359 | - | then [BooleanEntry(keyIsActive, true)] | |
360 | - | else throw("Strict value is not equal to itself.") | |
361 | - | } | |
362 | - | ||
363 | - | ||
364 | - | ||
365 | - | @Callable(i) | |
366 | - | func stopBridge () = { | |
367 | - | let callerCheck = assertCallerIsManager(i, STOP_MANAGER) | |
368 | - | if ((callerCheck == callerCheck)) | |
369 | - | then [BooleanEntry(keyIsActive, false)] | |
370 | - | else throw("Strict value is not equal to itself.") | |
371 | - | } | |
372 | - | ||
373 | - | ||
374 | - | ||
375 | - | @Callable(i) | |
376 | - | func setMinFee (assetId,minFee) = { | |
377 | - | let callerCheck = assertCallerIsManager(i, ASSET_MANAGER) | |
378 | - | if ((callerCheck == callerCheck)) | |
107 | + | func createLock (lockId,recipient,amount,lockDestination,assetSourceAndAddress) = { | |
108 | + | let lockIdStr = toBase64String(lockId) | |
109 | + | let lockRecipientKey = (lockIdStr + keyLockRecipient) | |
110 | + | let checkCaller = assertCallerIsBridge(i) | |
111 | + | if ((checkCaller == checkCaller)) | |
379 | 112 | then { | |
380 | - | let assetIdStr = toBase64String(assetId) | |
381 | - | [IntegerEntry((assetIdStr + keyAssetMinFee), minFee)] | |
113 | + | let checkLockId = assertValidLockId(lockId) | |
114 | + | if ((checkLockId == checkLockId)) | |
115 | + | then { | |
116 | + | let checkLock = assertLockNotExists(lockRecipientKey) | |
117 | + | if ((checkLock == checkLock)) | |
118 | + | then [BinaryEntry((lockIdStr + keyLockRecipient), recipient), IntegerEntry((lockIdStr + keyLockAmount), amount), BinaryEntry((lockIdStr + keyLockDestination), lockDestination), BinaryEntry((lockIdStr + keyLockAssetSource), assetSourceAndAddress)] | |
119 | + | else throw("Strict value is not equal to itself.") | |
120 | + | } | |
121 | + | else throw("Strict value is not equal to itself.") | |
382 | 122 | } | |
383 | 123 | else throw("Strict value is not equal to itself.") | |
384 | 124 | } | |
386 | 126 | ||
387 | 127 | ||
388 | 128 | @Callable(i) | |
389 | - | func | |
390 | - | let | |
391 | - | if (( | |
392 | - | then [ | |
129 | + | func setAdmin (newAdmin) = { | |
130 | + | let checkCaller = assertCallerIsAdmin(i) | |
131 | + | if ((checkCaller == checkCaller)) | |
132 | + | then [BinaryEntry(keyAdmin, newAdmin)] | |
393 | 133 | else throw("Strict value is not equal to itself.") | |
394 | 134 | } | |
395 | 135 | ||
396 | 136 | ||
397 | 137 | ||
398 | 138 | @Callable(i) | |
399 | - | func setAssetState (assetId,state) = { | |
400 | - | let callerCheck = assertCallerIsManager(i, ASSET_MANAGER) | |
401 | - | if ((callerCheck == callerCheck)) | |
402 | - | then { | |
403 | - | let assetIdStr = toBase64String(assetId) | |
404 | - | [BooleanEntry((assetIdStr + keyAssetIsActive), state)] | |
405 | - | } | |
139 | + | func setOracle (newOracle) = { | |
140 | + | let checkCaller = assertCallerIsAdmin(i) | |
141 | + | if ((checkCaller == checkCaller)) | |
142 | + | then [BinaryEntry(keyOracle, newOracle)] | |
143 | + | else throw("Strict value is not equal to itself.") | |
144 | + | } | |
145 | + | ||
146 | + | ||
147 | + | ||
148 | + | @Callable(i) | |
149 | + | func setBridge (newBridge) = { | |
150 | + | let checkCaller = assertCallerIsAdmin(i) | |
151 | + | if ((checkCaller == checkCaller)) | |
152 | + | then [BinaryEntry(keyBridge, newBridge)] | |
406 | 153 | else throw("Strict value is not equal to itself.") | |
407 | 154 | } | |
408 | 155 |
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 5 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | - | let | |
4 | + | let keyLockRecipient = "_lr" | |
5 | 5 | ||
6 | - | let | |
6 | + | let keyLockAmount = "_la" | |
7 | 7 | ||
8 | - | let | |
8 | + | let keyLockDestination = "_ld" | |
9 | 9 | ||
10 | - | let | |
10 | + | let keyLockAssetSource = "_las" | |
11 | 11 | ||
12 | - | let | |
12 | + | let keyVersion = "_v" | |
13 | 13 | ||
14 | - | let | |
14 | + | let keyBridge = "_b" | |
15 | 15 | ||
16 | - | let | |
16 | + | let keyOracle = "_o" | |
17 | 17 | ||
18 | - | let | |
18 | + | let keyAdmin = "_a" | |
19 | 19 | ||
20 | - | let keyUnlockSigner = "_us" | |
21 | - | ||
22 | - | let keyIsActive = "_ia" | |
23 | - | ||
24 | - | let keyManager = "_m" | |
25 | - | ||
26 | - | let BRIDGE_MANAGER = "BRIDGE_MANAGER" | |
27 | - | ||
28 | - | let ASSET_MANAGER = "ASSET_MANAGER" | |
29 | - | ||
30 | - | let STOP_MANAGER = "STOP_MANAGER" | |
31 | - | ||
32 | - | let baseAssetSourceAndAddress = base58'fRpRFUxiLXbzAaAT3sfi1oTFa8n4X8b9FUaQHyT5MkdXYpGZV' | |
33 | - | ||
34 | - | let baseAssetId = base58'6scFjhFGDfpmYySMKQ9vDbZuH8aMRWsUQJAHXzm1FsJo' | |
35 | - | ||
36 | - | let chainWaves = base58'3EMsPJ' | |
37 | - | ||
38 | - | let BP = 10000 | |
39 | - | ||
40 | - | let systemPrecision = 9 | |
20 | + | let keyUnlock = "_u" | |
41 | 21 | ||
42 | 22 | let errUnauthorized = "unauthorized" | |
43 | 23 | ||
44 | 24 | let errUninitialized = "uninitialized" | |
45 | 25 | ||
46 | - | let | |
26 | + | let errInvalidSignature = "invalid signature" | |
47 | 27 | ||
48 | - | let | |
28 | + | let errAlreadyClaimed = "claimed" | |
49 | 29 | ||
50 | - | let | |
30 | + | let errAlreadyLocked = "locked" | |
51 | 31 | ||
52 | - | let | |
32 | + | let errInvalidLockId = "invalid lockId" | |
53 | 33 | ||
54 | - | let | |
34 | + | let chainWaves = "V0FWRQ==" | |
55 | 35 | ||
56 | - | let errAssetNotFound = "asset not found" | |
57 | - | ||
58 | - | let errNotEnoughBalance = "not enough balance" | |
59 | - | ||
60 | - | let errBigPowValue = "pow value too big" | |
61 | - | ||
62 | - | let errBridgeDisabled = "bridge is disabled" | |
63 | - | ||
64 | - | let errAssetDisabled = "asset is disabled" | |
65 | - | ||
66 | - | let errWrongDestinationChain = "wrong destination chain" | |
67 | - | ||
68 | - | let typeBase = 0 | |
69 | - | ||
70 | - | let typeNative = 1 | |
71 | - | ||
72 | - | let typeWrapped = 2 | |
73 | - | ||
74 | - | func assertCallerIsManager (i,managerType) = match getBinary((managerType + keyManager)) { | |
36 | + | func assertCaller (i,key) = match getBinary(key) { | |
75 | 37 | case authority: ByteVector => | |
76 | 38 | if ((authority != i.caller.bytes)) | |
77 | 39 | then throw(errUnauthorized) | |
78 | 40 | else unit | |
79 | 41 | case _ => | |
80 | 42 | throw(errUninitialized) | |
81 | 43 | } | |
82 | 44 | ||
83 | 45 | ||
84 | - | func assertBridgeIsActive () = if (!(valueOrElse(getBoolean(keyIsActive), false))) | |
85 | - | then throw(errBridgeDisabled) | |
86 | - | else unit | |
46 | + | func assertCallerIsBridge (i) = assertCaller(i, keyBridge) | |
87 | 47 | ||
88 | 48 | ||
89 | - | func assertAssetIsActive (assetIdStr) = match getBoolean((assetIdStr + keyAssetIsActive)) { | |
90 | - | case isActive: Boolean => | |
91 | - | if (isActive) | |
92 | - | then unit | |
93 | - | else throw(errAssetDisabled) | |
49 | + | func assertCallerIsAdmin (i) = assertCaller(i, keyAdmin) | |
50 | + | ||
51 | + | ||
52 | + | func assertValidLockId (lockId) = { | |
53 | + | let version = valueOrErrorMessage(getBinary(keyVersion), errUninitialized) | |
54 | + | if (if ((size(lockId) != 16)) | |
55 | + | then true | |
56 | + | else (take(lockId, 1) != version)) | |
57 | + | then throw(errInvalidLockId) | |
58 | + | else unit | |
59 | + | } | |
60 | + | ||
61 | + | ||
62 | + | func assertLockNotExists (lockRecipientKey) = match getBinary(lockRecipientKey) { | |
63 | + | case t: ByteVector => | |
64 | + | throw(errAlreadyLocked) | |
94 | 65 | case _ => | |
95 | - | | |
66 | + | unit | |
96 | 67 | } | |
97 | 68 | ||
98 | 69 | ||
99 | - | func getFee (amount,assetIdStr) = { | |
100 | - | let baseFeeRateBP = valueOrErrorMessage(getInteger(keyBaseFeeRateBP), errUninitialized) | |
101 | - | let minFee = valueOrErrorMessage(getInteger((assetIdStr + keyAssetMinFee)), errAssetNotFound) | |
102 | - | let fee = ((amount * baseFeeRateBP) / BP) | |
103 | - | if ((minFee > fee)) | |
104 | - | then minFee | |
105 | - | else fee | |
106 | - | } | |
107 | - | ||
108 | - | ||
109 | - | func pow10 (value,power) = { | |
110 | - | let absPow = if ((0 > power)) | |
111 | - | then -(power) | |
112 | - | else power | |
113 | - | let powerValue = if ((absPow == 0)) | |
114 | - | then 1 | |
115 | - | else if ((absPow == 1)) | |
116 | - | then 10 | |
117 | - | else if ((absPow == 2)) | |
118 | - | then 100 | |
119 | - | else if ((absPow == 3)) | |
120 | - | then 1000 | |
121 | - | else if ((absPow == 4)) | |
122 | - | then 10000 | |
123 | - | else if ((absPow == 5)) | |
124 | - | then 100000 | |
125 | - | else if ((absPow == 6)) | |
126 | - | then 1000000 | |
127 | - | else if ((absPow == 7)) | |
128 | - | then 10000000 | |
129 | - | else if ((absPow == 8)) | |
130 | - | then 100000000 | |
131 | - | else if ((absPow == 9)) | |
132 | - | then 1000000000 | |
133 | - | else throw(errBigPowValue) | |
134 | - | if ((0 > power)) | |
135 | - | then (value / powerValue) | |
136 | - | else (value * powerValue) | |
137 | - | } | |
138 | - | ||
139 | - | ||
140 | - | func toSystemPrecision (amount,precision) = pow10(amount, (systemPrecision - precision)) | |
141 | - | ||
142 | - | ||
143 | - | func fromSystemPrecision (amount,precision) = pow10(amount, (precision - systemPrecision)) | |
144 | - | ||
145 | - | ||
146 | 70 | @Callable(i) | |
147 | - | func init (admin, | |
148 | - | then throw( | |
149 | - | else [BinaryEntry( | |
71 | + | func init (admin,version,bridge,oracle) = if (isDefined(getBinary(keyAdmin))) | |
72 | + | then throw(errUnauthorized) | |
73 | + | else [BinaryEntry(keyVersion, version), BinaryEntry(keyBridge, bridge), BinaryEntry(keyOracle, oracle), BinaryEntry(keyAdmin, admin)] | |
150 | 74 | ||
151 | 75 | ||
152 | 76 | ||
153 | 77 | @Callable(i) | |
154 | - | func | |
155 | - | let | |
156 | - | if (( | |
78 | + | func createUnlock (lockId,recipient,amount,lockSource,tokenSourceAndAddress,signature) = { | |
79 | + | let checkLockId = assertValidLockId(lockId) | |
80 | + | if ((checkLockId == checkLockId)) | |
157 | 81 | then { | |
158 | - | let $t041684568 = if ((assetId == baseAssetId)) | |
159 | - | then if ((assetSourceAndAddress != baseAssetSourceAndAddress)) | |
160 | - | then throw(errInvalidValues) | |
161 | - | else $Tuple2(typeBase, 8) | |
162 | - | else { | |
163 | - | let asset = valueOrErrorMessage(assetInfo(assetId), errAssetNotFound) | |
164 | - | let type = if ((asset.issuer == this)) | |
165 | - | then typeWrapped | |
166 | - | else typeNative | |
167 | - | $Tuple2(type, asset.decimals) | |
168 | - | } | |
169 | - | let type = $t041684568._1 | |
170 | - | let precision = $t041684568._2 | |
171 | - | let assetIdStr = toBase64String(assetId) | |
172 | - | let keySourceAddress = (toBase64String(assetSourceAndAddress) + keyAssetAddress) | |
173 | - | let keyNativeAddress = (assetIdStr + keyAssetAddress) | |
174 | - | if (if (isDefined(getBinary(keySourceAddress))) | |
175 | - | then true | |
176 | - | else isDefined(getBinary(keyNativeAddress))) | |
177 | - | then throw(errAlreadyExists) | |
178 | - | else [BinaryEntry(keySourceAddress, assetId), BinaryEntry(keyNativeAddress, assetSourceAndAddress), IntegerEntry((assetIdStr + keyAssetType), type), IntegerEntry((assetIdStr + keyAssetPrecision), precision), IntegerEntry((assetIdStr + keyAssetMinFee), minFee), BooleanEntry((assetIdStr + keyAssetIsActive), true)] | |
179 | - | } | |
180 | - | else throw("Strict value is not equal to itself.") | |
181 | - | } | |
182 | - | ||
183 | - | ||
184 | - | ||
185 | - | @Callable(i) | |
186 | - | func issue (name,description,precision) = { | |
187 | - | let callerCheck = assertCallerIsManager(i, ASSET_MANAGER) | |
188 | - | if ((callerCheck == callerCheck)) | |
189 | - | then [Issue(name, description, 0, precision, true)] | |
190 | - | else throw("Strict value is not equal to itself.") | |
191 | - | } | |
192 | - | ||
193 | - | ||
194 | - | ||
195 | - | @Callable(i) | |
196 | - | func removeAsset (assetSourceAndAddress,newAuthority) = { | |
197 | - | let callerCheck = assertCallerIsManager(i, ASSET_MANAGER) | |
198 | - | if ((callerCheck == callerCheck)) | |
199 | - | then { | |
200 | - | let keySourceAddress = (toBase64String(assetSourceAndAddress) + keyAssetAddress) | |
201 | - | let assetId = valueOrErrorMessage(getBinary(keySourceAddress), errNotExists) | |
202 | - | let assetIdStr = toBase64String(assetId) | |
203 | - | let type = valueOrErrorMessage(getInteger((assetIdStr + keyAssetType)), errNotExists) | |
204 | - | let actions = if ((type == typeBase)) | |
82 | + | let checkCaller = assertCallerIsBridge(i) | |
83 | + | if ((checkCaller == checkCaller)) | |
205 | 84 | then { | |
206 | - | let balance = wavesBalance(this) | |
207 | - | if ((balance.available > 0)) | |
208 | - | then [ScriptTransfer(Address(newAuthority), balance.available, unit)] | |
209 | - | else nil | |
210 | - | } | |
211 | - | else if ((type == typeNative)) | |
212 | - | then { | |
213 | - | let balance = assetBalance(this, assetId) | |
214 | - | if ((balance > 0)) | |
215 | - | then [ScriptTransfer(Address(newAuthority), balance, assetId)] | |
216 | - | else nil | |
217 | - | } | |
218 | - | else if ((type == typeWrapped)) | |
219 | - | then nil | |
220 | - | else throw(errInvalidValues) | |
221 | - | (actions ++ [DeleteEntry(keySourceAddress), DeleteEntry((assetIdStr + keyAssetAddress)), DeleteEntry((assetIdStr + keyAssetType)), DeleteEntry((assetIdStr + keyAssetPrecision)), DeleteEntry((assetIdStr + keyAssetMinFee)), DeleteEntry((assetIdStr + keyAssetIsActive))]) | |
222 | - | } | |
223 | - | else throw("Strict value is not equal to itself.") | |
224 | - | } | |
225 | - | ||
226 | - | ||
227 | - | ||
228 | - | @Callable(i) | |
229 | - | func lock (lockId,recipient,destination) = { | |
230 | - | let birdgeCheck = assertBridgeIsActive() | |
231 | - | if ((birdgeCheck == birdgeCheck)) | |
232 | - | then if ((destination == chainWaves)) | |
233 | - | then throw(errWrongDestinationChain) | |
234 | - | else if ((size(i.payments) != 1)) | |
235 | - | then throw(errNotOnePayment) | |
236 | - | else { | |
237 | - | let assetId = valueOrElse(i.payments[0].assetId, baseAssetId) | |
238 | - | let assetIdStr = toBase64String(assetId) | |
239 | - | let assetCheck = assertAssetIsActive(assetIdStr) | |
240 | - | if ((assetCheck == assetCheck)) | |
241 | - | then { | |
242 | - | let amount = i.payments[0].amount | |
243 | - | let validatorAddress = valueOrErrorMessage(getBinary(keyValidator), errUninitialized) | |
244 | - | let fee = getFee(amount, assetIdStr) | |
245 | - | let amountWithoutFee = (amount - fee) | |
246 | - | if ((0 >= amountWithoutFee)) | |
247 | - | then throw(errNotEnoughBalance) | |
248 | - | else { | |
249 | - | let assetSourceAndAddress = valueOrErrorMessage(getBinary((assetIdStr + keyAssetAddress)), errAssetNotFound) | |
250 | - | let type = valueOrErrorMessage(getInteger((assetIdStr + keyAssetType)), errAssetNotFound) | |
251 | - | let precision = valueOrErrorMessage(getInteger((assetIdStr + keyAssetPrecision)), errAssetNotFound) | |
252 | - | let feeCollector = valueOrErrorMessage(getBinary(keyFeeCollector), errUninitialized) | |
253 | - | let createLock = invoke(Address(validatorAddress), "createLock", [lockId, recipient, toSystemPrecision(amountWithoutFee, precision), destination, assetSourceAndAddress], nil) | |
254 | - | if ((createLock == createLock)) | |
255 | - | then if ((type == typeBase)) | |
256 | - | then [ScriptTransfer(Address(feeCollector), fee, unit)] | |
257 | - | else if ((type == typeNative)) | |
258 | - | then [ScriptTransfer(Address(feeCollector), fee, assetId)] | |
259 | - | else if ((type == typeWrapped)) | |
260 | - | then [Burn(assetId, amountWithoutFee), ScriptTransfer(Address(feeCollector), fee, assetId)] | |
261 | - | else throw(errInvalidValues) | |
262 | - | else throw("Strict value is not equal to itself.") | |
263 | - | } | |
264 | - | } | |
265 | - | else throw("Strict value is not equal to itself.") | |
266 | - | } | |
267 | - | else throw("Strict value is not equal to itself.") | |
268 | - | } | |
269 | - | ||
270 | - | ||
271 | - | ||
272 | - | @Callable(i) | |
273 | - | func unlock (lockId,recipient,amount,lockSource,assetSourceAndAddress,signature) = { | |
274 | - | let birdgeCheck = assertBridgeIsActive() | |
275 | - | if ((birdgeCheck == birdgeCheck)) | |
276 | - | then { | |
277 | - | let assetId = valueOrErrorMessage(getBinary((toBase64String(assetSourceAndAddress) + keyAssetAddress)), errAssetNotFound) | |
278 | - | let assetIdStr = toBase64String(assetId) | |
279 | - | let type = valueOrErrorMessage(getInteger((assetIdStr + keyAssetType)), errAssetNotFound) | |
280 | - | let precision = valueOrErrorMessage(getInteger((assetIdStr + keyAssetPrecision)), errAssetNotFound) | |
281 | - | let validatorAddress = valueOrErrorMessage(getBinary(keyValidator), errUninitialized) | |
282 | - | let unlockSigner = valueOrErrorMessage(getBinary(keyUnlockSigner), errUninitialized) | |
283 | - | let feeCollector = valueOrErrorMessage(getBinary(keyFeeCollector), errUninitialized) | |
284 | - | let createUnlock = invoke(Address(validatorAddress), "createUnlock", [lockId, recipient, amount, lockSource, assetSourceAndAddress, signature], nil) | |
285 | - | if ((createUnlock == createUnlock)) | |
286 | - | then { | |
287 | - | let isValid = match createUnlock { | |
288 | - | case v: Boolean => | |
289 | - | v | |
290 | - | case _ => | |
291 | - | false | |
292 | - | } | |
293 | - | if (!(isValid)) | |
294 | - | then throw(errInvalidValues) | |
85 | + | let lockIdStr = toBase64String(lockId) | |
86 | + | let lockSourceStr = toBase64String(lockSource) | |
87 | + | let unlockKey = (((lockSourceStr + "_") + lockIdStr) + keyUnlock) | |
88 | + | if (isDefined(getBoolean(unlockKey))) | |
89 | + | then throw(errAlreadyClaimed) | |
295 | 90 | else { | |
296 | - | let fee = if ((unlockSigner == i.caller.bytes)) | |
297 | - | then valueOrErrorMessage(getInteger((assetIdStr + keyAssetMinFee)), errAssetNotFound) | |
298 | - | else 0 | |
299 | - | let amountToSend = fromSystemPrecision(amount, precision) | |
300 | - | let amountToSendWithoutFee = (amountToSend - fee) | |
301 | - | if ((0 >= amountToSendWithoutFee)) | |
302 | - | then throw(errNotEnoughBalance) | |
303 | - | else if ((type == typeBase)) | |
304 | - | then ([ScriptTransfer(Address(recipient), amountToSendWithoutFee, unit)] ++ (if ((fee > 0)) | |
305 | - | then [ScriptTransfer(Address(feeCollector), fee, unit)] | |
306 | - | else nil)) | |
307 | - | else if ((type == typeNative)) | |
308 | - | then ([ScriptTransfer(Address(recipient), amountToSendWithoutFee, assetId)] ++ (if ((fee > 0)) | |
309 | - | then [ScriptTransfer(Address(feeCollector), fee, assetId)] | |
310 | - | else nil)) | |
311 | - | else if ((type == typeWrapped)) | |
312 | - | then ([Reissue(assetId, amountToSend, true), ScriptTransfer(Address(recipient), amountToSendWithoutFee, assetId)] ++ (if ((fee > 0)) | |
313 | - | then [ScriptTransfer(Address(feeCollector), fee, assetId)] | |
314 | - | else nil)) | |
315 | - | else throw(errInvalidValues) | |
91 | + | let message = makeString([lockIdStr, toBase64String(recipient), toString(amount), lockSourceStr, toBase64String(tokenSourceAndAddress), chainWaves], "_") | |
92 | + | let hash = keccak256(toBytes(message)) | |
93 | + | let recoveredKey = ecrecover(hash, signature) | |
94 | + | if ((recoveredKey != getBinaryValue(keyOracle))) | |
95 | + | then throw(errInvalidSignature) | |
96 | + | else $Tuple2([BooleanEntry(unlockKey, true)], true) | |
316 | 97 | } | |
317 | 98 | } | |
318 | 99 | else throw("Strict value is not equal to itself.") | |
319 | 100 | } | |
320 | 101 | else throw("Strict value is not equal to itself.") | |
321 | 102 | } | |
322 | 103 | ||
323 | 104 | ||
324 | 105 | ||
325 | 106 | @Callable(i) | |
326 | - | func setManager (managerType,manager) = { | |
327 | - | let callerCheck = assertCallerIsManager(i, BRIDGE_MANAGER) | |
328 | - | if ((callerCheck == callerCheck)) | |
329 | - | then [BinaryEntry((managerType + keyManager), manager)] | |
330 | - | else throw("Strict value is not equal to itself.") | |
331 | - | } | |
332 | - | ||
333 | - | ||
334 | - | ||
335 | - | @Callable(i) | |
336 | - | func setFeeCollector (feeCollector) = { | |
337 | - | let callerCheck = assertCallerIsManager(i, BRIDGE_MANAGER) | |
338 | - | if ((callerCheck == callerCheck)) | |
339 | - | then [BinaryEntry(keyFeeCollector, feeCollector)] | |
340 | - | else throw("Strict value is not equal to itself.") | |
341 | - | } | |
342 | - | ||
343 | - | ||
344 | - | ||
345 | - | @Callable(i) | |
346 | - | func setValidator (validator) = { | |
347 | - | let callerCheck = assertCallerIsManager(i, BRIDGE_MANAGER) | |
348 | - | if ((callerCheck == callerCheck)) | |
349 | - | then [BinaryEntry(keyValidator, validator)] | |
350 | - | else throw("Strict value is not equal to itself.") | |
351 | - | } | |
352 | - | ||
353 | - | ||
354 | - | ||
355 | - | @Callable(i) | |
356 | - | func startBridge () = { | |
357 | - | let callerCheck = assertCallerIsManager(i, BRIDGE_MANAGER) | |
358 | - | if ((callerCheck == callerCheck)) | |
359 | - | then [BooleanEntry(keyIsActive, true)] | |
360 | - | else throw("Strict value is not equal to itself.") | |
361 | - | } | |
362 | - | ||
363 | - | ||
364 | - | ||
365 | - | @Callable(i) | |
366 | - | func stopBridge () = { | |
367 | - | let callerCheck = assertCallerIsManager(i, STOP_MANAGER) | |
368 | - | if ((callerCheck == callerCheck)) | |
369 | - | then [BooleanEntry(keyIsActive, false)] | |
370 | - | else throw("Strict value is not equal to itself.") | |
371 | - | } | |
372 | - | ||
373 | - | ||
374 | - | ||
375 | - | @Callable(i) | |
376 | - | func setMinFee (assetId,minFee) = { | |
377 | - | let callerCheck = assertCallerIsManager(i, ASSET_MANAGER) | |
378 | - | if ((callerCheck == callerCheck)) | |
107 | + | func createLock (lockId,recipient,amount,lockDestination,assetSourceAndAddress) = { | |
108 | + | let lockIdStr = toBase64String(lockId) | |
109 | + | let lockRecipientKey = (lockIdStr + keyLockRecipient) | |
110 | + | let checkCaller = assertCallerIsBridge(i) | |
111 | + | if ((checkCaller == checkCaller)) | |
379 | 112 | then { | |
380 | - | let assetIdStr = toBase64String(assetId) | |
381 | - | [IntegerEntry((assetIdStr + keyAssetMinFee), minFee)] | |
113 | + | let checkLockId = assertValidLockId(lockId) | |
114 | + | if ((checkLockId == checkLockId)) | |
115 | + | then { | |
116 | + | let checkLock = assertLockNotExists(lockRecipientKey) | |
117 | + | if ((checkLock == checkLock)) | |
118 | + | then [BinaryEntry((lockIdStr + keyLockRecipient), recipient), IntegerEntry((lockIdStr + keyLockAmount), amount), BinaryEntry((lockIdStr + keyLockDestination), lockDestination), BinaryEntry((lockIdStr + keyLockAssetSource), assetSourceAndAddress)] | |
119 | + | else throw("Strict value is not equal to itself.") | |
120 | + | } | |
121 | + | else throw("Strict value is not equal to itself.") | |
382 | 122 | } | |
383 | 123 | else throw("Strict value is not equal to itself.") | |
384 | 124 | } | |
385 | 125 | ||
386 | 126 | ||
387 | 127 | ||
388 | 128 | @Callable(i) | |
389 | - | func | |
390 | - | let | |
391 | - | if (( | |
392 | - | then [ | |
129 | + | func setAdmin (newAdmin) = { | |
130 | + | let checkCaller = assertCallerIsAdmin(i) | |
131 | + | if ((checkCaller == checkCaller)) | |
132 | + | then [BinaryEntry(keyAdmin, newAdmin)] | |
393 | 133 | else throw("Strict value is not equal to itself.") | |
394 | 134 | } | |
395 | 135 | ||
396 | 136 | ||
397 | 137 | ||
398 | 138 | @Callable(i) | |
399 | - | func setAssetState (assetId,state) = { | |
400 | - | let callerCheck = assertCallerIsManager(i, ASSET_MANAGER) | |
401 | - | if ((callerCheck == callerCheck)) | |
402 | - | then { | |
403 | - | let assetIdStr = toBase64String(assetId) | |
404 | - | [BooleanEntry((assetIdStr + keyAssetIsActive), state)] | |
405 | - | } | |
139 | + | func setOracle (newOracle) = { | |
140 | + | let checkCaller = assertCallerIsAdmin(i) | |
141 | + | if ((checkCaller == checkCaller)) | |
142 | + | then [BinaryEntry(keyOracle, newOracle)] | |
143 | + | else throw("Strict value is not equal to itself.") | |
144 | + | } | |
145 | + | ||
146 | + | ||
147 | + | ||
148 | + | @Callable(i) | |
149 | + | func setBridge (newBridge) = { | |
150 | + | let checkCaller = assertCallerIsAdmin(i) | |
151 | + | if ((checkCaller == checkCaller)) | |
152 | + | then [BinaryEntry(keyBridge, newBridge)] | |
406 | 153 | else throw("Strict value is not equal to itself.") | |
407 | 154 | } | |
408 | 155 | ||
409 | 156 |
github/deemru/w8io/169f3d6 46.22 ms ◑