-
Notifications
You must be signed in to change notification settings - Fork 479
/
Semantics.purs
1219 lines (1004 loc) · 54.8 KB
/
Semantics.purs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
module Semantics where
import Prelude
import Data.BigInteger (BigInteger, fromInt)
import Data.FoldableWithIndex (foldrWithIndexDefault)
import Data.Foldable (foldMap)
import Data.List (List(Nil, Cons), concat, foldl, foldr)
import Data.Maybe (Maybe(..), fromMaybe)
import Data.Monoid (mempty)
import Data.Newtype (unwrap)
import Data.Tuple (Tuple(..))
import Marlowe.Types (BlockNumber, Choice, ValueF(..), ObservationF(..), ContractF(..), Contract(Use, Let, Scale, While, When, Choice, Both, Pay, Commit, Null), IdAction, IdChoice, IdCommit, IdOracle, LetLabel, Observation(FalseObs, TrueObs, ValueEQ, ValueLE, ValueLT, ValueGT, ValueGE, ChoseSomething, ChoseThis, NotObs, OrObs, AndObs, BelowTimeout), Person, Timeout, Value(ValueFromOracle, ValueFromChoice, ModValue, DivValue, MulValue, SubValue, AddValue, NegValue, Constant, Committed, CurrentBlock), WIdChoice(WIdChoice))
import Matryoshka (Algebra, cata)
import Data.Foldable as F
import Data.Map as M
import Data.Set as S
-- Data type for Inputs with their information
data Input
= IChoice IdChoice Choice
| IOracle IdOracle BlockNumber BigInteger
derive instance eqInput :: Eq Input
derive instance ordInput :: Ord Input
data AnyInput
= Action IdAction
| Input Input
derive instance eqAnyInput :: Eq AnyInput
derive instance ordAnyInput :: Ord AnyInput
data IdInput
= InputIdChoice IdChoice
| IdOracle IdOracle
derive instance eqIdInput :: Eq IdInput
derive instance ordIdInput :: Ord IdInput
type TimeoutData
= M.Map Timeout (S.Set IdCommit)
type CommitInfoRecord
= {person :: Person, amount :: BigInteger, timeout :: Timeout}
data CommitInfo
= CommitInfo {redeemedPerPerson :: M.Map Person BigInteger, currentCommitsById :: M.Map IdCommit CommitInfoRecord, expiredCommitIds :: S.Set IdCommit, timeoutData :: TimeoutData}
type OracleDataPoint
= {blockNumber :: BlockNumber, value :: BigInteger}
data State
= State {commits :: CommitInfo, choices :: M.Map WIdChoice Choice, oracles :: M.Map IdOracle OracleDataPoint, usedIds :: S.Set IdAction}
emptyCommitInfo :: CommitInfo
emptyCommitInfo = CommitInfo { redeemedPerPerson: M.empty
, currentCommitsById: M.empty
, expiredCommitIds: S.empty
, timeoutData: M.empty
}
emptyState :: State
emptyState = State { commits: emptyCommitInfo
, choices: M.empty
, oracles: M.empty
, usedIds: S.empty
}
-- Adds a commit identifier to the timeout data map
addToCommByTim ::
Timeout ->
IdCommit ->
TimeoutData ->
TimeoutData
addToCommByTim timeout idCommit timData = case M.lookup timeout timData of
Just commSet -> M.insert timeout (S.insert idCommit commSet) timData
Nothing -> M.insert timeout (S.singleton idCommit) timData
-- Remove a commit identifier from the timeout data map
removeFromCommByTim ::
Timeout ->
IdCommit ->
TimeoutData ->
TimeoutData
removeFromCommByTim timeout idCommit timData = case M.lookup timeout timData of
Just commSet -> let newCommSet = S.delete idCommit commSet
in if S.isEmpty newCommSet
then M.delete timeout timData
else M.insert timeout newCommSet timData
Nothing -> timData
-- Add a commit to CommitInfo
addCommit ::
IdCommit ->
Person ->
BigInteger ->
Timeout ->
State ->
State
addCommit idCommit person value timeout (State state) = State (state { commits = CommitInfo (ci { currentCommitsById = M.insert idCommit ({ person
, amount: value
, timeout
}) ci.currentCommitsById, timeoutData = addToCommByTim timeout idCommit ci.timeoutData }) })
where
(CommitInfo ci) = state.commits
-- Return the person corresponding to a commit
personForCurrentCommit ::
IdCommit ->
State ->
Maybe Person
personForCurrentCommit idCommit (State state) = case M.lookup idCommit ci.currentCommitsById of
Just v -> Just v.person
Nothing -> Nothing
where
(CommitInfo ci) = state.commits
-- Check whether the commit is current (committed not expired)
isCurrentCommit ::
IdCommit ->
State ->
Boolean
isCurrentCommit idCommit (State { commits: (CommitInfo ci) }) = M.member idCommit ci.currentCommitsById
-- Check whether the commit is expired
isExpiredCommit ::
IdCommit ->
State ->
Boolean
isExpiredCommit idCommit (State state) = idCommit `S.member` ci.expiredCommitIds
where
CommitInfo ci = state.commits
-- Remove a current commit from CommitInfo
removeCurrentCommit ::
IdCommit ->
State ->
State
removeCurrentCommit idCommit (State state) = case M.lookup idCommit commById of
Just v -> State (state { commits = CommitInfo (ci { currentCommitsById = M.delete idCommit commById, timeoutData = removeFromCommByTim v.timeout idCommit timData }) })
Nothing -> State state
where
CommitInfo ci = state.commits
commById = ci.currentCommitsById
timData = ci.timeoutData
-- Get expired not collected for person
getRedeemedForPersonCI ::
Person ->
CommitInfo ->
BigInteger
getRedeemedForPersonCI person (CommitInfo ci) = fromMaybe (fromInt 0) (M.lookup person ci.redeemedPerPerson)
getRedeemedForPerson :: Person -> State -> BigInteger
getRedeemedForPerson person (State state) = getRedeemedForPersonCI person state.commits
-- Set the amount in redeemedPerPerson to zero
resetRedeemedForPerson ::
Person ->
State ->
State
resetRedeemedForPerson person (State state) = State (state { commits = CommitInfo (ci { redeemedPerPerson = M.delete person rppm }) })
where
(CommitInfo ci) = state.commits
rppm = ci.redeemedPerPerson
discountAvailableMoneyFromCommit ::
IdCommit ->
BigInteger ->
State ->
Maybe State
discountAvailableMoneyFromCommit idCommit discount (State state) = case M.lookup idCommit commById of
Just { person, amount, timeout } -> Just (State (state { commits = CommitInfo (ci { currentCommitsById = M.insert idCommit ({ person
, amount: amount - discount
, timeout
}) commById }) }))
Nothing -> Nothing
where
CommitInfo ci = state.commits
commById = ci.currentCommitsById
getAvailableAmountInCommit :: IdCommit -> State -> BigInteger
getAvailableAmountInCommit idCommit (State state) = case M.lookup idCommit ci.currentCommitsById of
Just v -> v.amount
Nothing -> fromInt 0
where
CommitInfo ci = state.commits
-- Collect inputs needed by a contract
collectNeededInputsFromValue ::
Value ->
S.Set IdInput
collectNeededInputsFromValue (CurrentBlock) = S.empty
collectNeededInputsFromValue (Committed _) = S.empty
collectNeededInputsFromValue (Constant _) = S.empty
collectNeededInputsFromValue (NegValue value) = collectNeededInputsFromValue value
collectNeededInputsFromValue (AddValue value1 value2) = S.unions [ collectNeededInputsFromValue value1
, collectNeededInputsFromValue value2
]
collectNeededInputsFromValue (SubValue value1 value2) = S.unions [ collectNeededInputsFromValue value1
, collectNeededInputsFromValue value2
]
collectNeededInputsFromValue (MulValue value1 value2) = S.unions [ collectNeededInputsFromValue value1
, collectNeededInputsFromValue value2
]
collectNeededInputsFromValue (DivValue value1 value2 value3) = S.unions [ collectNeededInputsFromValue value1
, collectNeededInputsFromValue value2
, collectNeededInputsFromValue value3
]
collectNeededInputsFromValue (ModValue value1 value2 value3) = S.unions [ collectNeededInputsFromValue value1
, collectNeededInputsFromValue value2
, collectNeededInputsFromValue value3
]
collectNeededInputsFromValue (ValueFromChoice idChoice value) = S.unions [ S.singleton (InputIdChoice idChoice)
, collectNeededInputsFromValue value
]
collectNeededInputsFromValue (ValueFromOracle idOracle value) = S.unions [ S.singleton (IdOracle idOracle)
, collectNeededInputsFromValue value
]
collectNeededInputsFromObservation :: Observation -> S.Set IdInput
collectNeededInputsFromObservation (BelowTimeout _) = S.empty
collectNeededInputsFromObservation (AndObs observation1 observation2) = S.unions [ collectNeededInputsFromObservation observation1
, collectNeededInputsFromObservation observation2
]
collectNeededInputsFromObservation (OrObs observation1 observation2) = S.unions [ collectNeededInputsFromObservation observation1
, collectNeededInputsFromObservation observation2
]
collectNeededInputsFromObservation (NotObs observation) = collectNeededInputsFromObservation observation
collectNeededInputsFromObservation (ChoseThis idChoice _) = S.singleton (InputIdChoice idChoice)
collectNeededInputsFromObservation (ChoseSomething idChoice) = S.singleton (InputIdChoice idChoice)
collectNeededInputsFromObservation (ValueGE value1 value2) = S.unions [ collectNeededInputsFromValue value1
, collectNeededInputsFromValue value2
]
collectNeededInputsFromObservation (ValueGT value1 value2) = S.unions [ collectNeededInputsFromValue value1
, collectNeededInputsFromValue value2
]
collectNeededInputsFromObservation (ValueLT value1 value2) = S.unions [ collectNeededInputsFromValue value1
, collectNeededInputsFromValue value2
]
collectNeededInputsFromObservation (ValueLE value1 value2) = S.unions [ collectNeededInputsFromValue value1
, collectNeededInputsFromValue value2
]
collectNeededInputsFromObservation (ValueEQ value1 value2) = S.unions [ collectNeededInputsFromValue value1
, collectNeededInputsFromValue value2
]
collectNeededInputsFromObservation (TrueObs) = S.empty
collectNeededInputsFromObservation (FalseObs) = S.empty
collectNeededInputsFromContract :: Contract -> S.Set IdInput
collectNeededInputsFromContract Null = S.empty
collectNeededInputsFromContract (Commit _ _ _ value _ _ contract1 contract2) = S.unions [ collectNeededInputsFromValue value
, collectNeededInputsFromContract contract1
, collectNeededInputsFromContract contract2
]
collectNeededInputsFromContract (Pay _ _ _ value _ contract1 contract2) = S.unions [ collectNeededInputsFromValue value
, collectNeededInputsFromContract contract1
, collectNeededInputsFromContract contract2
]
collectNeededInputsFromContract (Both contract1 contract2) = S.unions [ collectNeededInputsFromContract contract1
, collectNeededInputsFromContract contract2
]
collectNeededInputsFromContract (Choice observation contract1 contract2) = S.unions [ collectNeededInputsFromObservation observation
, collectNeededInputsFromContract contract1
, collectNeededInputsFromContract contract2
]
collectNeededInputsFromContract (When observation _ contract1 contract2) = S.unions [ collectNeededInputsFromObservation observation
, collectNeededInputsFromContract contract1
, collectNeededInputsFromContract contract2
]
collectNeededInputsFromContract (While observation _ contract1 contract2) = S.unions [ collectNeededInputsFromObservation observation
, collectNeededInputsFromContract contract1
, collectNeededInputsFromContract contract2
]
collectNeededInputsFromContract (Scale value1 value2 value3 contract) = S.unions [ collectNeededInputsFromValue value1
, collectNeededInputsFromValue value2
, collectNeededInputsFromValue value3
, collectNeededInputsFromContract contract
]
collectNeededInputsFromContract (Let _ contract1 contract2) = S.unions [ collectNeededInputsFromContract contract1
, collectNeededInputsFromContract contract2
]
collectNeededInputsFromContract (Use _) = S.empty
-- Add inputs and action ids to state.
-- Return Nothing on redundant or irrelevant inputs
addAnyInput ::
BlockNumber ->
AnyInput ->
S.Set IdInput ->
State ->
Maybe State
-- current block ---^
addAnyInput _ (Action idInput) _ (State state)
| idInput `S.member` state.usedIds = Nothing
| true = Just (State (state { usedIds = S.insert idInput usedIdsSet }))
where
usedIdsSet = state.usedIds
addAnyInput _ (Input (IChoice idChoice choice)) neededInputs (State state)
| (WIdChoice idChoice) `M.member` state.choices = Nothing
| (InputIdChoice idChoice) `S.member` neededInputs = Just (State (state { choices = M.insert (WIdChoice idChoice) choice state.choices }))
| true = Nothing
addAnyInput blockNumber (Input (IOracle idOracle timestamp value)) neededInputs (State state) = case M.lookup idOracle oracleMap of
Just v -> if (timestamp > v.blockNumber) && (timestamp <= blockNumber) && ((IdOracle idOracle) `S.member` neededInputs)
then Just newState
else Nothing
Nothing -> Just newState
where
newState = State (state { oracles = M.insert idOracle { blockNumber: timestamp
, value
} oracleMap
})
oracleMap = state.oracles
-- Decides whether something has expired
isExpired ::
BlockNumber ->
BlockNumber ->
Boolean
isExpired currBlockNum expirationBlockNum = currBlockNum >= expirationBlockNum
-- Expire commits
expireOneCommit ::
IdCommit ->
CommitInfo ->
CommitInfo
expireOneCommit idCommit (CommitInfo commitInfo) = CommitInfo case M.lookup idCommit currentCommits of
Just { person, amount } -> let redeemedBefore = case M.lookup person redPerPer of
Just x -> x
Nothing -> fromInt 0
in commitInfo { redeemedPerPerson = M.insert person (redeemedBefore + amount) redPerPer, currentCommitsById = M.delete idCommit currentCommits }
Nothing -> commitInfo
where
redPerPer = commitInfo.redeemedPerPerson
currentCommits = commitInfo.currentCommitsById
expireManyCommits :: S.Set IdCommit -> CommitInfo -> CommitInfo
expireManyCommits commitIds (CommitInfo commitInfo) = foldr expireOneCommit semiUpdatedCI commitIds
where
semiUpdatedCI = CommitInfo (commitInfo { expiredCommitIds = S.union expiComms commitIds
})
expiComms = commitInfo.expiredCommitIds
expireCommitsCI :: BlockNumber -> CommitInfo -> CommitInfo
expireCommitsCI blockNumber (CommitInfo commitInfo) = case M.findMin timData of
Just res -> let minBlock = res.key
commIds = res.value
remTimData = M.delete minBlock timData
in if isExpired blockNumber minBlock
then let partUpdatedCommitInfo = CommitInfo (commitInfo { timeoutData = remTimData
})
updatedCommitInfo = expireManyCommits commIds partUpdatedCommitInfo
in expireCommitsCI blockNumber updatedCommitInfo
else CommitInfo commitInfo
Nothing -> CommitInfo commitInfo
where
timData = commitInfo.timeoutData
expireCommits :: BlockNumber -> State -> State
expireCommits blockNumber (State state) = State (state { commits = expireCommitsCI blockNumber commitInfo })
where
commitInfo = state.commits
-- Evaluate a value
evalValue ::
BlockNumber ->
State ->
Value ->
BigInteger
evalValue blockNumber _ CurrentBlock = blockNumber
evalValue _ state (Committed idCommit) = getAvailableAmountInCommit idCommit state
evalValue _ _ (Constant value) = value
evalValue blockNumber state (NegValue value) = -(evalValue blockNumber state value)
evalValue blockNumber state (AddValue lhs rhs) = (go lhs) + (go rhs)
where
go = evalValue blockNumber state
evalValue blockNumber state (SubValue lhs rhs) = (go lhs) - (go rhs)
where
go = evalValue blockNumber state
evalValue blockNumber state (MulValue lhs rhs) = (go lhs) * (go rhs)
where
go = evalValue blockNumber state
evalValue blockNumber state (DivValue dividend divisor defaultVal) = if actualDivisor == (fromInt 0)
then go defaultVal
else div (go dividend) actualDivisor
where
go = evalValue blockNumber state
actualDivisor = go divisor
evalValue blockNumber state (ModValue dividend divisor defaultVal) = if actualDivisor == (fromInt 0)
then go defaultVal
else mod (go dividend) actualDivisor
where
go = evalValue blockNumber state
actualDivisor = go divisor
evalValue blockNumber (State state) (ValueFromChoice idChoice val) = fromMaybe (evalValue blockNumber (State state) val) (M.lookup (WIdChoice idChoice) state.choices)
evalValue blockNumber (State state) (ValueFromOracle idOracle val) = case M.lookup idOracle state.oracles of
Just v -> v.value
Nothing -> evalValue blockNumber (State state) val
-- Evaluate an observation
evalObservation ::
BlockNumber ->
State ->
Observation ->
Boolean
evalObservation blockNumber _ (BelowTimeout timeout) = not (isExpired blockNumber timeout)
evalObservation blockNumber state (AndObs obs1 obs2) = (go obs1) && (go obs2)
where
go = evalObservation blockNumber state
evalObservation blockNumber state (OrObs obs1 obs2) = (go obs1) || (go obs2)
where
go = evalObservation blockNumber state
evalObservation blockNumber state (NotObs obs) = not (evalObservation blockNumber state obs)
evalObservation _ (State state) (ChoseThis idChoice choice) = case M.lookup (WIdChoice idChoice) (state.choices) of
Just actualChoice -> actualChoice == choice
Nothing -> false
evalObservation _ (State state) (ChoseSomething idChoice) = (WIdChoice idChoice) `M.member` (state.choices)
evalObservation blockNumber state (ValueGE val1 val2) = (go val1) >= (go val2)
where
go = evalValue blockNumber state
evalObservation blockNumber state (ValueGT val1 val2) = (go val1) > (go val2)
where
go = evalValue blockNumber state
evalObservation blockNumber state (ValueLT val1 val2) = (go val1) < (go val2)
where
go = evalValue blockNumber state
evalObservation blockNumber state (ValueLE val1 val2) = (go val1) <= (go val2)
where
go = evalValue blockNumber state
evalObservation blockNumber state (ValueEQ val1 val2) = (go val1) == (go val2)
where
go = evalValue blockNumber state
evalObservation _ _ TrueObs = true
evalObservation _ _ FalseObs = false
isNormalised :: BigInteger -> BigInteger -> Boolean
isNormalised divid divis = divid == divis && divid /= (fromInt 0)
type Environment
= M.Map BigInteger Contract
emptyEnvironment :: Environment
emptyEnvironment = M.empty
addToEnvironment :: LetLabel -> Contract -> Environment -> Environment
addToEnvironment = M.insert
lookupEnvironment :: LetLabel -> Environment -> Maybe Contract
lookupEnvironment = M.lookup
-- Find the highest label in the given Contract (assuming there is an order to LetLabels)
maxIdFromContract ::
Contract ->
LetLabel
maxIdFromContract Null = (fromInt 0)
maxIdFromContract (Commit _ _ _ _ _ _ contract1 contract2) = (max (maxIdFromContract contract1) (maxIdFromContract contract2))
maxIdFromContract (Pay _ _ _ _ _ contract1 contract2) = (max (maxIdFromContract contract1) (maxIdFromContract contract2))
maxIdFromContract (Both contract1 contract2) = (max (maxIdFromContract contract1) (maxIdFromContract contract2))
maxIdFromContract (Choice _ contract1 contract2) = (max (maxIdFromContract contract1) (maxIdFromContract contract2))
maxIdFromContract (When _ _ contract1 contract2) = (max (maxIdFromContract contract1) (maxIdFromContract contract2))
maxIdFromContract (While _ _ contract1 contract2) = (max (maxIdFromContract contract1) (maxIdFromContract contract2))
maxIdFromContract (Scale _ _ _ contract) = (maxIdFromContract contract)
maxIdFromContract (Let letLabel contract1 contract2) = max letLabel (max (maxIdFromContract contract1) (maxIdFromContract contract2))
maxIdFromContract (Use letLabel) = letLabel
-- Looks for an unused label in the Environment and Contract provided
-- (assuming that labels are numbers)
getFreshLabel ::
Environment ->
Contract ->
LetLabel
getFreshLabel env c = (fromInt 1) + (max (case M.findMax env of
Nothing -> fromInt 0
Just v -> v.key) (maxIdFromContract c))
-- We check subcontract in case it uses undefined labels
-- (to prevent potential infinite loop)
-- Optimisation: We do not need to check existing bindings because all
-- contracts are nullified before being added to Environment;
-- Ensures all Use are defined in Environment, if they are not
-- they are replaced with Null
nullifyInvalidUses ::
Environment ->
Contract ->
Contract
nullifyInvalidUses _ Null = Null
nullifyInvalidUses env (Commit idAction idCommit person value timeout1 timeout2 contract1 contract2) = Commit idAction idCommit person value timeout1 timeout2 (nullifyInvalidUses env contract1) (nullifyInvalidUses env contract2)
nullifyInvalidUses env (Pay idAction idCommit person value timeout contract1 contract2) = Pay idAction idCommit person value timeout (nullifyInvalidUses env contract1) (nullifyInvalidUses env contract2)
nullifyInvalidUses env (Both contract1 contract2) = Both (nullifyInvalidUses env contract1) (nullifyInvalidUses env contract2)
nullifyInvalidUses env (Choice observation contract1 contract2) = Choice observation (nullifyInvalidUses env contract1) (nullifyInvalidUses env contract2)
nullifyInvalidUses env (When observation timeout contract1 contract2) = When observation timeout (nullifyInvalidUses env contract1) (nullifyInvalidUses env contract2)
nullifyInvalidUses env (While observation timeout contract1 contract2) = While observation timeout (nullifyInvalidUses env contract1) (nullifyInvalidUses env contract2)
nullifyInvalidUses env (Scale value1 value2 value3 contract) = Scale value1 value2 value3 (nullifyInvalidUses env contract)
nullifyInvalidUses env (Let letLabel contract1 contract2) = Let letLabel (nullifyInvalidUses env contract1) (nullifyInvalidUses newEnv contract2)
where
newEnv = addToEnvironment letLabel Null env
-- We just need to mark it as available for this function
nullifyInvalidUses env (Use letLabel) = case lookupEnvironment letLabel env of
Nothing -> Null
Just _ -> Use letLabel
-- Replaces a label with another one (unless it is shadowed)
relabel ::
LetLabel ->
LetLabel ->
Contract ->
Contract
relabel _ _ Null = Null
relabel startLabel endLabel (Commit idAction idCommit person value timeout1 timeout2 contract1 contract2) = Commit idAction idCommit person value timeout1 timeout2 (relabel startLabel endLabel contract1) (relabel startLabel endLabel contract2)
relabel startLabel endLabel (Pay idAction idCommit person value timeout contract1 contract2) = Pay idAction idCommit person value timeout (relabel startLabel endLabel contract1) (relabel startLabel endLabel contract2)
relabel startLabel endLabel (Both contract1 contract2) = Both (relabel startLabel endLabel contract1) (relabel startLabel endLabel contract2)
relabel startLabel endLabel (Choice observation contract1 contract2) = Choice observation (relabel startLabel endLabel contract1) (relabel startLabel endLabel contract2)
relabel startLabel endLabel (When observation timeout contract1 contract2) = When observation timeout (relabel startLabel endLabel contract1) (relabel startLabel endLabel contract2)
relabel startLabel endLabel (While observation timeout contract1 contract2) = While observation timeout (relabel startLabel endLabel contract1) (relabel startLabel endLabel contract2)
relabel startLabel endLabel (Scale value1 value2 value3 contract) = Scale value1 value2 value3 (relabel startLabel endLabel contract)
relabel startLabel endLabel (Let letLabel contract1 contract2) = Let letLabel (relabel startLabel endLabel contract1) (if (letLabel == startLabel)
then contract2
else relabel startLabel endLabel contract2)
relabel startLabel endLabel (Use letLabel) = if (letLabel == startLabel)
then Use endLabel
else Use letLabel
-- Reduce non actionable primitives and remove expired
reduceRec ::
BlockNumber ->
State ->
Environment ->
Contract ->
Contract
reduceRec _ _ _ Null = Null
reduceRec blockNum state env c@(Commit _ _ _ _ timeout_start timeout_end _ continuation) = if (isExpired blockNum timeout_start) || (isExpired blockNum timeout_end)
then reduceRec blockNum state env continuation
else c
reduceRec blockNum state env c@(Pay _ _ _ _ timeout _ continuation) = if isExpired blockNum timeout
then reduceRec blockNum state env continuation
else c
reduceRec blockNum state env (Both cont1 cont2) = Both (go cont1) (go cont2)
where
go = reduceRec blockNum state env
reduceRec blockNum state env (Choice obs cont1 cont2) = reduceRec blockNum state env (if (evalObservation blockNum state obs)
then cont1
else cont2)
reduceRec blockNum state env c@(When obs timeout cont1 cont2) = if isExpired blockNum timeout
then go cont2
else if evalObservation blockNum state obs
then go cont1
else c
where
go = reduceRec blockNum state env
reduceRec blockNum state env (Scale divid divis def contract) = Scale (Constant vsDivid) (Constant vsDivis) (Constant vsDef) (go contract)
where
go = reduceRec blockNum state env
vsDivid = evalValue blockNum state divid
vsDivis = evalValue blockNum state divis
vsDef = evalValue blockNum state def
reduceRec blockNum state env (While obs timeout contractWhile contractAfter) = if isExpired blockNum timeout
then go contractAfter
else if evalObservation blockNum state obs
then (While obs timeout (go contractWhile) contractAfter)
else go contractAfter
where
go = reduceRec blockNum state env
reduceRec blockNum state env (Let label boundContract contract) = case lookupEnvironment label env of
Nothing -> let newEnv = addToEnvironment label checkedBoundContract env
in Let label checkedBoundContract (reduceRec blockNum state newEnv contract)
Just _ -> let freshLabel = getFreshLabel env contract
newEnv = addToEnvironment freshLabel checkedBoundContract env
fixedContract = relabel label freshLabel contract
in Let freshLabel checkedBoundContract (reduceRec blockNum state newEnv fixedContract)
where
checkedBoundContract = nullifyInvalidUses env boundContract
reduceRec blockNum state env (Use label) = case lookupEnvironment label env of
Just contract -> reduceRec blockNum state env contract
Nothing -> Null
-- Optimisation: We do not need to restore environment of the binding because:
-- * We ensure entries are not overwritten in Environment
-- * We check that all entries of Environment had their Use defined when added
reduce ::
BlockNumber ->
State ->
Contract ->
Contract
reduce blockNum state contract = reduceRec blockNum state emptyEnvironment contract
-- Reduce useless primitives to Null
simplify_aux ::
Contract ->
{contract :: Contract, uses :: S.Set LetLabel}
simplify_aux Null = { contract: Null, uses: S.empty }
simplify_aux (Commit idAction idCommit person value timeout1 timeout2 contract1 contract2) = let v1 = simplify_aux contract1
v2 = simplify_aux contract2
contract = Commit idAction idCommit person value timeout1 timeout2 v1.contract v2.contract
uses = S.union (v1.uses) (v2.uses)
in { contract
, uses
}
simplify_aux (Pay idAction idCommit person value timeout contract1 contract2) = let v1 = simplify_aux contract1
v2 = simplify_aux contract2
contract = Pay idAction idCommit person value timeout v1.contract v2.contract
uses = S.union (v1.uses) (v2.uses)
in { contract
, uses
}
simplify_aux (Both contract1 contract2) = case simplify_aux contract1, simplify_aux contract2 of
v1@{ contract: Null }, v2 -> v2
v1, v2@{ contract: Null } -> v1
v1, v2 -> { contract: Both v1.contract v2.contract
, uses: S.union v1.uses v2.uses
}
simplify_aux (Choice observation contract1 contract2) = let v1 = simplify_aux contract1
v2 = simplify_aux contract2
contract = if (v1.contract == Null) && (v2.contract == Null)
then Null
else Choice observation v1.contract v2.contract
uses = S.union v1.uses v2.uses
in { contract, uses }
simplify_aux (When observation timeout contract1 contract2) = let v1 = simplify_aux contract1
v2 = simplify_aux contract2
contract = if (v1.contract == Null) && (v2.contract == Null)
then Null
else When observation timeout v1.contract v2.contract
uses = S.union v1.uses v2.uses
in { contract
, uses
}
simplify_aux (While observation timeout contract1 contract2) = let v1 = simplify_aux contract1
v2 = simplify_aux contract2
in if (v1.contract == Null) && (v2.contract == Null)
then { contract: Null
, uses: S.empty
}
else { contract: While observation timeout v1.contract v2.contract
, uses: S.union v1.uses v2.uses
}
simplify_aux (Scale value1 value2 value3 contract) = let v = simplify_aux contract
in if (v.contract == Null)
then { contract: Null
, uses: S.empty
}
else { contract: Scale value1 value2 value3 v.contract
, uses: v.uses
}
simplify_aux (Let letLabel contract1 contract2) = (let v1 = simplify_aux contract1
v2 = simplify_aux contract2
in (if S.member letLabel v2.uses
then { contract: Let letLabel v1.contract v2.contract
, uses: S.union v1.uses (S.delete letLabel v2.uses)
}
else { contract: v2.contract
, uses: v2.uses
}))
simplify_aux (Use letLabel) = { contract: Use letLabel
, uses: S.singleton letLabel
}
simplify :: Contract -> Contract
simplify c = let v = simplify_aux c
in v.contract
-- How much everybody pays or receives in transaction
type TransactionOutcomes
= M.Map Person BigInteger
emptyOutcome :: TransactionOutcomes
emptyOutcome = M.empty
isEmptyOutcome :: TransactionOutcomes -> Boolean
isEmptyOutcome trOut = F.all (\x ->
x == (fromInt 0)) trOut
-- Adds a value to the map of outcomes
addOutcome ::
Person ->
BigInteger ->
TransactionOutcomes ->
TransactionOutcomes
addOutcome person diffValue trOut = M.insert person newValue trOut
where
newValue = case M.lookup person trOut of
Just value -> value + diffValue
Nothing -> diffValue
-- Get effect of outcomes on the bank of the contract
outcomeEffect ::
TransactionOutcomes ->
BigInteger
outcomeEffect trOut = foldl (-) (fromInt 0) trOut
-- Add two transaction outcomes together
combineOutcomes ::
TransactionOutcomes ->
TransactionOutcomes ->
TransactionOutcomes
combineOutcomes = M.unionWith (+)
data FetchResult a
= Picked a
| NoMatch
| MultipleMatches
data DetachedPrimitive
= DCommit IdCommit Person BigInteger Timeout
| DPay IdCommit Person BigInteger
derive instance eqDetachedPrimitive :: Eq DetachedPrimitive
derive instance ordDetachedPrimitive :: Ord DetachedPrimitive
-- Semantics of Scale
scaleValue ::
BigInteger ->
BigInteger ->
BigInteger ->
BigInteger ->
BigInteger
scaleValue divid divis def val = if (divis == fromInt 0)
then def
else ((val * divid) `div` divis)
scaleResult ::
BigInteger ->
BigInteger ->
BigInteger ->
DetachedPrimitive ->
DetachedPrimitive
scaleResult divid divis def (DCommit idCommit person val tim) = DCommit idCommit person (scaleValue divid divis def val) tim
scaleResult divid divis def (DPay idCommit person val) = DPay idCommit person (scaleValue divid divis def val)
-- Find out whether the Action is allowed given the current state
-- and contract, and, if so, pick the corresponding primitive in the contract.
-- Also return the contract without the selected primitive.
fetchPrimitive ::
IdAction ->
BlockNumber ->
State ->
Contract ->
FetchResult {prim :: DetachedPrimitive, contract :: Contract}
-- Remaining contract --^
fetchPrimitive idAction blockNum state (Commit idActionC idCommit person value _ timeout continuation _) = if (idAction == idActionC && notCurrentCommit && notExpiredCommit)
then Picked { prim: (DCommit idCommit person actualValue timeout)
, contract: continuation
}
else NoMatch
where
notCurrentCommit = not (isCurrentCommit idCommit state)
notExpiredCommit = not (isExpiredCommit idCommit state)
actualValue = evalValue blockNum state value
fetchPrimitive idAction blockNum state (Pay idActionC idCommit person value _ continuation _) = if (idAction == idActionC)
then Picked { prim: (DPay idCommit person actualValue)
, contract: continuation
}
else NoMatch
where
actualValue = evalValue blockNum state value
fetchPrimitive idAction blockNum state (Both leftContract rightContract) = case Tuple (go leftContract) (go rightContract) of
Tuple (Picked v) NoMatch -> Picked { prim: v.prim
, contract: Both v.contract rightContract
}
Tuple NoMatch (Picked v) -> Picked { prim: v.prim
, contract: Both leftContract v.contract
}
Tuple NoMatch NoMatch -> NoMatch
_ -> MultipleMatches
where
go = fetchPrimitive idAction blockNum state
fetchPrimitive idAction blockNum state (While obs timeout contract1 contract2) = case fetchPrimitive idAction blockNum state contract1 of
Picked v -> Picked { prim: v.prim
, contract: While obs timeout v.contract contract2
}
NoMatch -> NoMatch
MultipleMatches -> MultipleMatches
fetchPrimitive idAction blockNum state (Let label boundContract subContract) = case fetchPrimitive idAction blockNum state subContract of
Picked v -> Picked { prim: v.prim
, contract: Let label boundContract v.contract
}
NoMatch -> NoMatch
MultipleMatches -> MultipleMatches
fetchPrimitive idAction blockNum state (Scale divid divis def subContract) = case fetchPrimitive idAction blockNum state subContract of
Picked v -> Picked { prim: scaleResult sDivid sDivis sDef v.prim
, contract: Scale divid divis def v.contract
}
NoMatch -> NoMatch
MultipleMatches -> MultipleMatches
where
sDivid = evalValue blockNum state divid
sDivis = evalValue blockNum state divis
sDef = evalValue blockNum state def
fetchPrimitive _ _ _ _ = NoMatch
-- Gather all the primitives that are reachable given the current state of the contract
scoutPrimitivesAux :: BlockNumber -> State -> Contract -> M.Map IdAction (Maybe DetachedPrimitive)
scoutPrimitivesAux blockNum state (Commit idActionC idCommit person value _ timeout continuation _) =
if (notCurrentCommit && notExpiredCommit)
then M.insert idActionC (Just (DCommit idCommit person actualValue timeout)) M.empty
else M.empty
where
notCurrentCommit = not (isCurrentCommit idCommit state)
notExpiredCommit = not (isExpiredCommit idCommit state)
actualValue = evalValue blockNum state value
scoutPrimitivesAux blockNum state (Pay idActionC idCommit person value _ continuation _) =
M.insert idActionC (Just (DPay idCommit person actualValue)) M.empty
where
actualValue = evalValue blockNum state value
scoutPrimitivesAux blockNum state (Both leftContract rightContract) =
M.unionWith (\_ _ -> Nothing) leftC rightC
where
leftC = go leftContract
rightC = go rightContract
go = scoutPrimitivesAux blockNum state
scoutPrimitivesAux blockNum state (While obs timeout contract1 contract2) =
scoutPrimitivesAux blockNum state contract1
scoutPrimitivesAux blockNum state (Let label boundContract subContract) =
scoutPrimitivesAux blockNum state subContract
scoutPrimitivesAux blockNum state (Scale divid divis def subContract) =
map (\x -> (scaleResult sDivid sDivis sDef) <$> x) subC
where
subC = scoutPrimitivesAux blockNum state subContract
sDivid = evalValue blockNum state divid
sDivis = evalValue blockNum state divis
sDef = evalValue blockNum state def
scoutPrimitivesAux _ _ _ = M.empty
data DetachedPrimitiveWIA =
DWAICommit IdAction IdCommit BigInteger Timeout
| DWAIPay IdAction IdCommit BigInteger
addMaybePrimitive :: IdAction -> Maybe DetachedPrimitive -> M.Map Person (List DetachedPrimitiveWIA) -> M.Map Person (List DetachedPrimitiveWIA)
addMaybePrimitive _ Nothing x = x
addMaybePrimitive idAction (Just (DCommit idCommit person val timeout)) pmap =
M.insert person (Cons (DWAICommit idAction idCommit val timeout) plist) pmap
where
plist = case M.lookup person pmap of
Just l -> l
Nothing -> Nil
addMaybePrimitive idAction (Just (DPay idCommit person val)) pmap =
M.insert person (Cons (DWAIPay idAction idCommit val) plist) pmap
where
plist = case M.lookup person pmap of
Just l -> l
Nothing -> Nil
scoutPrimitives :: BlockNumber -> State -> Contract -> M.Map Person (List DetachedPrimitiveWIA)
scoutPrimitives blockNum state contract =
foldrWithIndexDefault addMaybePrimitive M.empty res
where res = scoutPrimitivesAux blockNum state contract
data DynamicProblem
= NoProblem
| CommitNotMade
| NotEnoughMoneyLeftInCommit
| CommitIsExpired
derive instance eqDynamicProblem :: Eq DynamicProblem
derive instance ordDynamicProblem :: Ord DynamicProblem
data EvaluationResult a
= Result a DynamicProblem
| InconsistentState
-- This should not happen when using fetchPrimitive result
-- Evaluation of actionable input
eval ::
DetachedPrimitive ->
State ->
EvaluationResult {outcome :: TransactionOutcomes, state :: State}
eval (DCommit idCommit person value timeout) state = if (isCurrentCommit idCommit state) || (isExpiredCommit idCommit state)
then InconsistentState
else Result { outcome: addOutcome person (-value) emptyOutcome
, state: addCommit idCommit person value timeout state
} NoProblem
eval (DPay idCommit person value) state = if (not (isCurrentCommit idCommit state))
then (if (not (isExpiredCommit idCommit state))
then Result { outcome: emptyOutcome, state } CommitNotMade
else Result { outcome: emptyOutcome, state } CommitIsExpired)
else (if value > maxValue
then case discountAvailableMoneyFromCommit idCommit maxValue state of
Just newState -> Result { outcome: addOutcome person maxValue emptyOutcome
, state: newState
} NotEnoughMoneyLeftInCommit
Nothing -> InconsistentState