This repository has been archived by the owner on Feb 15, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 31
/
Copy pathfa2_owner_hooks_lib.mligo
166 lines (148 loc) · 5.71 KB
/
fa2_owner_hooks_lib.mligo
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
#if !FA2_BEHAVIORS
#define FA2_BEHAVIORS
(**
Generic implementation of the permission logic for sender and receiver hooks.
Actual behavior is driven by a `permissions_descriptor`.
To be used in FA2 and/or FA2 permission transfer hook contract implementation
which supports sender/receiver hooks.
*)
#include "../fa2_interface.mligo"
#include "../fa2_errors.mligo"
#include "../fa2_permissions_descriptor.mligo"
type get_owners = transfer_descriptor -> (address option) list
type hook_entry_point = transfer_descriptor_param contract
type hook_result =
| Hook_entry_point of hook_entry_point
| Hook_undefined of string
type to_hook = address -> hook_result
(**
Extracts a set of unique `from_` or `to_` addresses from the transfer batch.
@param batch transfer batch
@param get_owner selector of `from_` or `to_` addresses from each individual `transfer_descriptor`
*)
let get_owners_from_batch (batch, get_owners : (transfer_descriptor list) * get_owners) : address set =
List.fold
(fun (acc, tx : (address set) * transfer_descriptor) ->
let owners = get_owners tx in
List.fold
(fun (acc, o: (address set) * (address option)) ->
match o with
| None -> acc
| Some a -> Set.add a acc
)
owners
acc
)
batch
(Set.empty : address set)
let validate_owner_hook (p, get_owners, to_hook, is_required :
transfer_descriptor_param * get_owners * to_hook * bool)
: hook_entry_point list =
let owners = get_owners_from_batch (p.batch, get_owners) in
Set.fold
(fun (eps, owner : (hook_entry_point list) * address) ->
match to_hook owner with
| Hook_entry_point h -> h :: eps
| Hook_undefined error ->
(* owner hook is not implemented by the target contract *)
if is_required
then (failwith error : hook_entry_point list) (* owner hook is required: fail *)
else eps (* owner hook is optional: skip it *)
)
owners ([] : hook_entry_point list)
let validate_owner(p, policy, get_owners, to_hook :
transfer_descriptor_param * owner_hook_policy * get_owners * to_hook)
: hook_entry_point list =
match policy with
| Owner_no_hook -> ([] : hook_entry_point list)
| Optional_owner_hook -> validate_owner_hook (p, get_owners, to_hook, false)
| Required_owner_hook -> validate_owner_hook (p, get_owners, to_hook, true)
(**
Given an address of the token receiver, tries to get an entrypoint for
`fa2_token_receiver` interface.
*)
let to_receiver_hook : to_hook = fun (a : address) ->
let c : hook_entry_point option =
Tezos.get_entrypoint_opt "%tokens_received" a in
match c with
| Some c -> Hook_entry_point c
| None -> Hook_undefined fa2_receiver_hook_undefined
(**
Create a list iof Tezos operations invoking all token receiver contracts that
implement `fa2_token_receiver` interface. Fail if specified `owner_hook_policy`
cannot be met.
*)
let validate_receivers (p, receiver_policy : transfer_descriptor_param * owner_hook_policy)
: hook_entry_point list =
let get_receivers : get_owners = fun (tx : transfer_descriptor) ->
List.map (fun (t : transfer_destination_descriptor) -> t.to_ )tx.txs in
validate_owner (p, receiver_policy, get_receivers, to_receiver_hook)
(**
Given an address of the token sender, tries to get an entrypoint for
`fa2_token_sender` interface.
*)
let to_sender_hook : to_hook = fun (a : address) ->
let c : hook_entry_point option =
Tezos.get_entrypoint_opt "%tokens_sent" a in
match c with
| Some c -> Hook_entry_point c
| None -> Hook_undefined fa2_sender_hook_undefined
(**
Create a list iof Tezos operations invoking all token sender contracts that
implement `fa2_token_sender` interface. Fail if specified `owner_hook_policy`
cannot be met.
*)
let validate_senders (p, sender_policy : transfer_descriptor_param * owner_hook_policy)
: hook_entry_point list =
let get_sender : get_owners = fun (tx : transfer_descriptor) -> [tx.from_] in
validate_owner (p, sender_policy, get_sender, to_sender_hook)
(**
Generate a list of Tezos operations invoking sender and receiver hooks according to
the policies defined by the permissions descriptor.
To be used in FA2 and/or FA2 transfer hook contract implementation which supports
sender/receiver hooks.
*)
let get_owner_transfer_hooks (p, descriptor : transfer_descriptor_param * permissions_descriptor)
: hook_entry_point list =
let sender_entries = validate_senders (p, descriptor.sender) in
let receiver_entries = validate_receivers (p, descriptor.receiver) in
(* merge two lists *)
List.fold
(fun (l, ep : (hook_entry_point list) * hook_entry_point) -> ep :: l)
receiver_entries sender_entries
let transfers_to_descriptors (txs : transfer list) : transfer_descriptor list =
List.map
(fun (tx : transfer) ->
let txs = List.map
(fun (dst : transfer_destination) ->
{
to_ = Some dst.to_;
token_id = dst.token_id;
amount = dst.amount;
}
) tx.txs in
{
from_ = Some tx.from_;
txs = txs;
}
) txs
let transfers_to_transfer_descriptor_param
(txs, operator : (transfer list) * address) : transfer_descriptor_param =
{
batch = transfers_to_descriptors txs;
operator = operator;
}
(**
Gets operations to call sender/receiver hook for the specified transfer and
permission descriptor
*)
let get_owner_hook_ops_for (tx_descriptor, pd
: transfer_descriptor_param * permissions_descriptor) : operation list =
let hook_calls = get_owner_transfer_hooks (tx_descriptor, pd) in
match hook_calls with
| [] -> ([] : operation list)
| _ ->
List.map (fun(call: hook_entry_point) ->
Tezos.transaction tx_descriptor 0mutez call)
hook_calls
#endif