| 1 | [ |
| 2 | "autonomous agent", |
| 3 | { |
| 4 | "init": "{ |
| 5 | $close_timeout = params.timeout; |
| 6 | $asset = params.asset; |
| 7 | $addressA = params.addressA; |
| 8 | $addressB = params.addressB; |
| 9 | $bFromA = (trigger.address == $addressA); |
| 10 | $bFromB = (trigger.address == $addressB); |
| 11 | $bFromParties = ($bFromA OR $bFromB); |
| 12 | if ($bFromParties) |
| 13 | $party = $bFromA ? 'A' : 'B'; |
| 14 | }", |
| 15 | "messages": { |
| 16 | "cases": [ |
| 17 | { |
| 18 | "if": "{ $bFromParties AND ($asset != 'base' AND trigger.output[[asset=$asset]] > 0 OR $asset == 'base' AND trigger.output[[asset=base]] > 10000)}", |
| 19 | "init": "{ |
| 20 | if (var['close_initiated_by']){ |
| 21 | $refused=1; |
| 22 | } else { |
| 23 | if (!var['period']) |
| 24 | $period = 1; |
| 25 | else |
| 26 | $period = var['period']; |
| 27 | } |
| 28 | }", |
| 29 | "messages": [ |
| 30 | { |
| 31 | "if": "{!$refused}", |
| 32 | "app": "data", |
| 33 | "payload": { |
| 34 | "open": 1, |
| 35 | "period": "{$period}", |
| 36 | "{$addressA}": "{var['balanceA'] + ($bFromA ? trigger.output[[asset=$asset]] : 0)}", |
| 37 | "{$addressB}": "{var['balanceB'] + ($bFromB ? trigger.output[[asset=$asset]] : 0)}", |
| 38 | "event_id": "{var['event_id'] otherwise 1}", |
| 39 | "trigger_unit": "{trigger.unit}" |
| 40 | } |
| 41 | }, |
| 42 | { |
| 43 | "if": "{$refused}", |
| 44 | "app": "data", |
| 45 | "payload": { |
| 46 | "refused": 1, |
| 47 | "trigger_unit": "{trigger.unit}", |
| 48 | "event_id": "{var['event_id'] otherwise 1}" |
| 49 | } |
| 50 | }, |
| 51 | { |
| 52 | "if": "{$refused}", |
| 53 | "app": "payment", |
| 54 | "payload": { |
| 55 | "asset": "{$asset}", |
| 56 | "outputs": [ |
| 57 | { |
| 58 | "address": "{trigger.address}", |
| 59 | "amount": "{$asset == 'base' ? (trigger.output[[asset=base]] - 10000) : trigger.output[[asset=$asset]]}" |
| 60 | } |
| 61 | ] |
| 62 | } |
| 63 | }, |
| 64 | { |
| 65 | "app": "state", |
| 66 | "state": "{ |
| 67 | if (!var['event_id']) |
| 68 | var['event_id'] = 2; |
| 69 | else |
| 70 | var['event_id'] += 1; |
| 71 | if (!$refused){ |
| 72 | if (!var['period']) |
| 73 | var['period'] = 1; |
| 74 | $key = 'balance' || $party; |
| 75 | var[$key] += trigger.output[[asset=$asset]]; |
| 76 | } |
| 77 | }" |
| 78 | } |
| 79 | ] |
| 80 | }, |
| 81 | { |
| 82 | "if": "{ $bFromParties AND trigger.data.close AND !var['close_initiated_by'] }", |
| 83 | "init": "{ |
| 84 | if (!is_integer(trigger.data.period) OR trigger.data.period != var['period']) |
| 85 | bounce('wrong period'); |
| 86 | $transferredFromMe = trigger.data.transferredFromMe otherwise 0; |
| 87 | if (!is_integer($transferredFromMe) OR $transferredFromMe < 0) |
| 88 | bounce('bad amount spent by me: ' || $transferredFromMe); |
| 89 | if (trigger.data.sentByPeer){ |
| 90 | if (trigger.data.sentByPeer.signed_message.aa_address != this_address) |
| 91 | bounce('signed for another channel'); |
| 92 | if (trigger.data.sentByPeer.signed_message.period != var['period']) |
| 93 | bounce('signed for a different period of this channel'); |
| 94 | if (!is_valid_signed_package(trigger.data.sentByPeer, $bFromB ? $addressA : $addressB)) |
| 95 | bounce('invalid signature by peer'); |
| 96 | $transferredFromPeer = trigger.data.sentByPeer.signed_message.amount_spent; |
| 97 | if (!is_integer($transferredFromPeer) OR $transferredFromPeer < 0) |
| 98 | bounce('bad amount spent by peer: ' || $transferredFromPeer); |
| 99 | } |
| 100 | else |
| 101 | $transferredFromPeer = 0; |
| 102 | }", |
| 103 | "messages": [ |
| 104 | { |
| 105 | "app": "data", |
| 106 | "payload": { |
| 107 | "closing": 1, |
| 108 | "period": "{var['period']}", |
| 109 | "initiated_by": "{trigger.address}", |
| 110 | "{$addressA}": "{ $bFromA ? $transferredFromMe : $transferredFromPeer}", |
| 111 | "{$addressB}": "{ $bFromB ? $transferredFromMe : $transferredFromPeer}", |
| 112 | "event_id": "{var['event_id'] otherwise 1}", |
| 113 | "trigger_unit": "{trigger.unit}" |
| 114 | } |
| 115 | }, |
| 116 | { |
| 117 | "app": "state", |
| 118 | "state": "{ |
| 119 | var['spentByA'] = $bFromA ? $transferredFromMe : $transferredFromPeer; |
| 120 | var['spentByB'] = $bFromB ? $transferredFromMe : $transferredFromPeer; |
| 121 | var['close_initiated_by'] = $party; |
| 122 | var['close_start_ts'] = timestamp; |
| 123 | if (!var['event_id']) |
| 124 | var['event_id'] = 2; |
| 125 | else |
| 126 | var['event_id'] += 1; |
| 127 | }" |
| 128 | } |
| 129 | ] |
| 130 | }, |
| 131 | { |
| 132 | "if": "{ trigger.data.confirm AND var['close_initiated_by'] }", |
| 133 | "init": "{ |
| 134 | if (!$bFromParties OR (var['close_initiated_by'] == $party AND timestamp < var['close_start_ts'] + $close_timeout)) |
| 135 | bounce('too early'); |
| 136 | if (!is_integer(trigger.data.period) OR trigger.data.period != var['period']) |
| 137 | bounce('wrong period'); |
| 138 | $additionalTransferredFromMe = trigger.data.additionalTransferredFromMe otherwise 0; |
| 139 | if (!is_integer($additionalTransferredFromMe) OR $additionalTransferredFromMe < 0) |
| 140 | bounce('bad additionalTransferredFromMe: ' || $additionalTransferredFromMe); |
| 141 | $additionalSpentByA = $party == 'A' ? $additionalTransferredFromMe : 0; |
| 142 | $additionalSpentByB = $party == 'B' ? $additionalTransferredFromMe : 0; |
| 143 | $balanceA = var['balanceA'] - var['spentByA'] + var['spentByB'] + $additionalSpentByB - $additionalSpentByA; |
| 144 | $balanceB = var['balanceB'] - var['spentByB'] + var['spentByA'] + $additionalSpentByA - $additionalSpentByB; |
| 145 | $finalBalanceA = $balanceA > 0 ? $balanceA : 0; |
| 146 | $finalBalanceB = $balanceB > 0 ? $balanceB : 0; |
| 147 | }", |
| 148 | "messages": [ |
| 149 | { |
| 150 | "if": "{$asset == 'base'}", |
| 151 | "app": "payment", |
| 152 | "payload": { |
| 153 | "asset": "base", |
| 154 | "outputs": [ |
| 155 | { |
| 156 | "address": "{$addressA}", |
| 157 | "amount": "{ $finalBalanceA < $finalBalanceB ? ($finalBalanceA + 10000) : '' }" |
| 158 | }, |
| 159 | { |
| 160 | "address": "{$addressB}", |
| 161 | "amount": "{ $finalBalanceA >= $finalBalanceB ? ($finalBalanceB + 10000) : '' }" |
| 162 | } |
| 163 | ] |
| 164 | } |
| 165 | }, |
| 166 | { |
| 167 | "if": "{$asset != 'base'}", |
| 168 | "app": "payment", |
| 169 | "payload": { |
| 170 | "asset": "base", |
| 171 | "outputs": [ |
| 172 | { |
| 173 | "address": "{$addressA}", |
| 174 | "amount": "{ $finalBalanceA < $finalBalanceB ? 10000 : '' }" |
| 175 | }, |
| 176 | { |
| 177 | "address": "{$addressB}", |
| 178 | "amount": "{ $finalBalanceA >= $finalBalanceB ? 10000 : '' }" |
| 179 | } |
| 180 | ] |
| 181 | } |
| 182 | }, |
| 183 | { |
| 184 | "if": "{$asset != 'base'}", |
| 185 | "app": "payment", |
| 186 | "payload": { |
| 187 | "asset": "{$asset}", |
| 188 | "outputs": [ |
| 189 | { |
| 190 | "address": "{$addressA}", |
| 191 | "amount": "{ $finalBalanceA < $finalBalanceB ? $finalBalanceA : '' }" |
| 192 | }, |
| 193 | { |
| 194 | "address": "{$addressB}", |
| 195 | "amount": "{ $finalBalanceA >= $finalBalanceB ? $finalBalanceB : '' }" |
| 196 | } |
| 197 | ] |
| 198 | } |
| 199 | }, |
| 200 | { |
| 201 | "app": "data", |
| 202 | "payload": { |
| 203 | "closed": 1, |
| 204 | "period": "{var['period']}", |
| 205 | "event_id": "{var['event_id']}" |
| 206 | } |
| 207 | }, |
| 208 | { |
| 209 | "app": "state", |
| 210 | "state": "{ |
| 211 | var['period'] += 1; |
| 212 | var['close_initiated_by'] = false; |
| 213 | var['close_start_ts'] = false; |
| 214 | var['balanceA'] = false; |
| 215 | var['balanceB'] = false; |
| 216 | var['spentByA'] = false; |
| 217 | var['spentByB'] = false; |
| 218 | var['event_id'] += 1; |
| 219 | }" |
| 220 | } |
| 221 | ] |
| 222 | }, |
| 223 | { |
| 224 | "if": "{ trigger.data.fraud_proof AND var['close_initiated_by'] AND trigger.data.sentByPeer }", |
| 225 | "init": "{ |
| 226 | $bInitiatedByA = (var['close_initiated_by'] == 'A'); |
| 227 | if (trigger.data.sentByPeer.signed_message.aa_address != this_address) |
| 228 | bounce('signed for another channel'); |
| 229 | if (trigger.data.sentByPeer.signed_message.period != var['period']) |
| 230 | bounce('signed for a different period of this channel'); |
| 231 | if (!is_valid_signed_package(trigger.data.sentByPeer, $bInitiatedByA ? $addressA : $addressB)) |
| 232 | bounce('invalid signature by peer'); |
| 233 | $transferredFromPeer = trigger.data.sentByPeer.signed_message.amount_spent; |
| 234 | if (!is_integer($transferredFromPeer) OR $transferredFromPeer < 0) |
| 235 | bounce('bad amount spent by peer: ' || $transferredFromPeer); |
| 236 | $transferredFromPeerAsClaimedByPeer = var['spentBy' || ($bInitiatedByA ? 'A' : 'B')]; |
| 237 | if ($transferredFromPeer <= $transferredFromPeerAsClaimedByPeer) |
| 238 | bounce("the peer didn't lie in his favor"); |
| 239 | }", |
| 240 | "messages": [ |
| 241 | { |
| 242 | "app": "payment", |
| 243 | "payload": { |
| 244 | "asset": "base", |
| 245 | "outputs": [ |
| 246 | { |
| 247 | "address": "{trigger.address}" |
| 248 | } |
| 249 | ] |
| 250 | } |
| 251 | }, |
| 252 | { |
| 253 | "if": "{$asset != 'base'}", |
| 254 | "app": "payment", |
| 255 | "payload": { |
| 256 | "asset": "{$asset}", |
| 257 | "outputs": [ |
| 258 | { |
| 259 | "address": "{trigger.address}" |
| 260 | } |
| 261 | ] |
| 262 | } |
| 263 | }, |
| 264 | { |
| 265 | "app": "data", |
| 266 | "payload": { |
| 267 | "closed": 1, |
| 268 | "fraud_proof": 1, |
| 269 | "period": "{var['period']}", |
| 270 | "event_id": "{var['event_id']}" |
| 271 | } |
| 272 | }, |
| 273 | { |
| 274 | "app": "state", |
| 275 | "state": "{ |
| 276 | var['period'] += 1; |
| 277 | var['close_initiated_by'] = false; |
| 278 | var['close_start_ts'] = false; |
| 279 | var['balanceA'] = false; |
| 280 | var['balanceB'] = false; |
| 281 | var['spentByA'] = false; |
| 282 | var['spentByB'] = false; |
| 283 | if (!var['event_id']) |
| 284 | var['event_id'] = 1; |
| 285 | else |
| 286 | var['event_id'] += 1; |
| 287 | }" |
| 288 | } |
| 289 | ] |
| 290 | } |
| 291 | ] |
| 292 | } |
| 293 | } |
| 294 | ] |