-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathsim.py
718 lines (601 loc) · 27.9 KB
/
sim.py
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
import asyncio
import time
import datetime
import pytimeparse
from pathlib import Path
from typing import Dict, List, Tuple, Optional, Union
from blspy import BasicSchemeMPL, PrivateKey, G1Element, AugSchemeMPL, G2Element
from chia.types.blockchain_format.sized_bytes import bytes32
from chia.types.blockchain_format.coin import Coin
from chia.types.blockchain_format.program import Program, SerializedProgram
from chia.types.spend_bundle import SpendBundle
from chia.types.coin_spend import CoinSpend
from chia.types.coin_record import CoinRecord
from chia.util.ints import uint64
from chia.util.condition_tools import ConditionOpcode
from chia.util.hash import std_hash
from chia.wallet.sign_coin_spends import sign_coin_spends
from chia.wallet.puzzles.p2_delegated_puzzle_or_hidden_puzzle import ( # standard_transaction
puzzle_for_pk,
calculate_synthetic_secret_key,
DEFAULT_HIDDEN_PUZZLE_HASH,
)
from clvm_tools.clvmc import compile_clvm
from chia.clvm.spend_sim import SpendSim, SimClient
from chia.consensus.default_constants import DEFAULT_CONSTANTS
duration_div = 86400.0
block_time = (600.0 / 32.0) / duration_div
# Allowed subdivisions of 1 coin
def secret_exponent_for_index(index: int) -> int:
blob = index.to_bytes(32, "big")
hashed_blob = BasicSchemeMPL.key_gen(std_hash(b"foo" + blob))
r = int.from_bytes(hashed_blob, "big")
return r
def private_key_for_index(index: int) -> PrivateKey:
r = secret_exponent_for_index(index)
return PrivateKey.from_bytes(r.to_bytes(32, "big"))
def public_key_for_index(index: int) -> G1Element:
return private_key_for_index(index).get_g1()
def sign_message_with_index(index: int, message: str) -> G2Element:
sk = private_key_for_index(index)
return AugSchemeMPL.sign(sk, bytes(message, "utf-8"))
def sign_messages_with_indexes(sign_ops: List[Dict[int, str]]) -> G2Element:
signatures = []
for _ in sign_ops:
for index, message in _.items():
sk = private_key_for_index(index)
signatures.append(AugSchemeMPL.sign(sk, bytes(message, "utf-8")))
return AugSchemeMPL.aggregate(signatures)
def aggregate_signatures(signatures: List[G2Element]) -> G2Element:
return AugSchemeMPL.aggregate(signatures)
class SpendResult:
def __init__(self, result: Dict):
"""Constructor for internal use.
error - a string describing the error or None
result - the raw result from Network::push_tx
outputs - a list of new Coin objects surviving the transaction
"""
self.result = result
if "error" in result:
self.error: Optional[str] = result["error"]
self.outputs: List[Coin] = []
else:
self.error = None
self.outputs = result["additions"]
def find_standard_coins(self, puzzle_hash: bytes32) -> List[Coin]:
"""Given a Wallet's puzzle_hash, find standard coins usable by it.
These coins are recognized as changing the Wallet's chia balance and are
usable for any purpose."""
return list(filter(lambda x: x.puzzle_hash == puzzle_hash, self.outputs))
class CoinWrapper(Coin):
"""A class that provides some useful methods on coins."""
def __init__(self, parent: Coin, puzzle_hash: bytes32, amt: uint64, source: Program):
"""Given parent, puzzle_hash and amount, give an object representing the coin"""
super().__init__(parent, puzzle_hash, amt)
self.source = source
def puzzle(self) -> Program:
"""Return the program that unlocks this coin"""
return self.source
def puzzle_hash(self) -> bytes32:
"""Return this coin's puzzle hash"""
return self.puzzle().get_tree_hash()
def smart_coin(self) -> "SmartCoinWrapper":
"""Return a smart coin object wrapping this coin's program"""
return SmartCoinWrapper(DEFAULT_CONSTANTS.GENESIS_CHALLENGE, self.source)
def as_coin(self) -> Coin:
return Coin(
self.parent_coin_info,
self.puzzle_hash,
self.amount,
)
@classmethod
def from_coin(cls, coin: Coin, puzzle: Program) -> "CoinWrapper":
return cls(
coin.parent_coin_info,
coin.puzzle_hash,
coin.amount,
puzzle,
)
def create_standard_spend(self, priv: PrivateKey, conditions: List[List]):
delegated_puzzle_solution = Program.to((1, conditions))
solution = Program.to([[], delegated_puzzle_solution, []])
coin_spend_object = CoinSpend(
self.as_coin(),
self.puzzle(),
solution,
)
# Create a signature for each of these. We'll aggregate them at the end.
signature: G2Element = AugSchemeMPL.sign(
calculate_synthetic_secret_key(priv, DEFAULT_HIDDEN_PUZZLE_HASH),
(delegated_puzzle_solution.get_tree_hash() + self.name() + DEFAULT_CONSTANTS.AGG_SIG_ME_ADDITIONAL_DATA),
)
return coin_spend_object, signature
# We have two cases for coins:
# - Wallet coins which contribute to the "wallet balance" of the user.
# They enable a user to "spend money" and "take actions on the network"
# that have monetary value.
#
# - Smart coins which either lock value or embody information and
# services. These also contain a chia balance but are used for purposes
# other than a fungible, liquid, spendable resource. They should not show
# up in a "wallet" in the same way. We should use them by locking value
# into wallet coins. We should ensure that value contained in a smart coin
# coin is never destroyed.
class SmartCoinWrapper:
def __init__(self, genesis_challenge: bytes32, source: Program):
"""A wrapper for a smart coin carrying useful methods for interacting with chia."""
self.genesis_challenge = genesis_challenge
self.source = source
def puzzle(self) -> Program:
"""Give this smart coin's program"""
return self.source
def puzzle_hash(self) -> bytes32:
"""Give this smart coin's puzzle hash"""
return self.source.get_tree_hash()
def custom_coin(self, parent: Coin, amt: uint64) -> CoinWrapper:
"""Given a parent and an amount, create the Coin object representing this
smart coin as it would exist post launch"""
return CoinWrapper(parent.name(), self.puzzle_hash(), amt, self.source)
# Used internally to accumulate a search for coins we can combine to the
# target amount.
# Result is the smallest set of coins whose sum of amounts is greater
# than target_amount.
class CoinPairSearch:
def __init__(self, target_amount: uint64):
self.target = target_amount
self.total: uint64 = uint64(0)
self.max_coins: List[Coin] = []
def get_result(self) -> Tuple[List[Coin], uint64]:
return self.max_coins, self.total
def insort(self, coin: Coin):
for i in range(len(self.max_coins)):
if self.max_coins[i].amount < coin.amount:
self.max_coins.insert(i, coin)
break
else:
self.max_coins.append(coin)
def process_coin_for_combine_search(self, coin: Coin):
self.total = uint64(self.total + coin.amount)
if len(self.max_coins) == 0:
self.max_coins.append(coin)
else:
self.insort(coin)
while (
(len(self.max_coins) > 0)
and (self.total - self.max_coins[-1].amount >= self.target)
and ((self.total - self.max_coins[-1].amount > 0) or (len(self.max_coins) > 1))
):
self.total = uint64(self.total - self.max_coins[-1].amount)
self.max_coins = self.max_coins[:-1]
# A basic wallet that knows about standard coins.
# We can use this to track our balance as an end user and keep track of
# chia that is released by smart coins, if the smart coins interact
# meaningfully with them, as many likely will.
class Wallet:
def __init__(self, parent: "Network", name: str, pk: G1Element, priv: PrivateKey):
"""Internal use constructor, use Network::make_wallet
Fields:
parent - The Network object that created this Wallet
name - The textural name of the actor
pk_ - The actor's public key
sk_ - The actor's private key
usable_coins - Standard coins spendable by this actor
puzzle - A program for creating this actor's standard coin
puzzle_hash - The puzzle hash for this actor's standard coin
pk_to_sk_dict - a dictionary for retrieving the secret keys when presented with the corresponding public key
"""
self.parent = parent
self.name = name
self.pk_ = pk
self.sk_ = priv
self.usable_coins: Dict[bytes32, Coin] = {}
self.puzzle: Program = puzzle_for_pk(self.pk())
self.puzzle_hash: bytes32 = self.puzzle.get_tree_hash()
synth_sk: PrivateKey = calculate_synthetic_secret_key(self.sk_, DEFAULT_HIDDEN_PUZZLE_HASH)
self.pk_to_sk_dict: Dict[str, PrivateKey] = {
str(self.pk_): self.sk_,
str(synth_sk.get_g1()): synth_sk,
}
def __repr__(self) -> str:
return f"<Wallet(name={self.name},puzzle_hash={self.puzzle_hash},pk={self.pk_})>"
# Make this coin available to the user it goes with.
def add_coin(self, coin: Coin):
self.usable_coins[coin.name()] = coin
def pk_to_sk(self, pk: G1Element) -> PrivateKey:
assert str(pk) in self.pk_to_sk_dict
return self.pk_to_sk_dict[str(pk)]
def compute_combine_action(
self, amt: uint64, actions: List, usable_coins: Dict[bytes32, Coin]
) -> Optional[List[Coin]]:
# No one coin is enough, try to find a best fit pair, otherwise combine the two
# maximum coins.
searcher = CoinPairSearch(amt)
# Process coins for this round.
for k, c in usable_coins.items():
searcher.process_coin_for_combine_search(c)
max_coins, total = searcher.get_result()
if total >= amt:
return max_coins
else:
return None
# Given some coins, combine them, causing the network to make us
# recompute our balance in return.
#
# This is a metaphor for the meaning of "solution" in the coin "puzzle program" context:
#
# The arguments to the coin's puzzle program, which is the first kind
# of object called a 'solution' in this case refers to the arguments
# needed to cause the program to emit blockchain compatible opcodes.
# It's a "puzzle" in the sense that a ctf RE exercise is a "puzzle".
# "solving" it means producing input that causes the unknown program to
# do something desirable.
#
# Spending multiple coins:
# There are two running lists that need to be kept when spending multiple coins:
#
# - A list of signatures. Each coin has its own signature requirements, but standard
# coins are signed like this:
#
# AugSchemeMPL.sign(
# calculate_synthetic_secret_key(self.sk_,DEFAULT_HIDDEN_PUZZLE_HASH),
# (delegated_puzzle_solution.get_tree_hash() + c.name() + DEFAULT_CONSTANTS.AGG_SIG_ME_ADDITIONAL_DATA)
# )
#
# where c.name is the coin's "name" (in the code) or coinID (in the
# chialisp docs). delegated_puzzle_solution is a clvm program that
# produces the conditions we want to give the puzzle program (the first
# kind of 'solution'), which will add the basic ones needed by owned
# standard coins.
#
# In most cases, we'll give a tuple like (1, [some, python, [data,
# here]]) for these arguments, because '1' is the quote function 'q' in
# clvm. One could write this program with any valid clvm code though.
# The important thing is that it's runnable code, not literal data as
# one might expect.
#
# - A list of CoinSpend objects.
# Theese consist of (with c : Coin):
#
# CoinSpend(
# c,
# c.puzzle(),
# solution,
# )
#
# Where solution is a second formulation of a 'solution' to a puzzle (the third form
# of 'solution' we'll use in this documentation is the CoinSpend object.). This is
# related to the program arguments above, but prefixes and suffixes an empty list on
# them (admittedly i'm cargo culting this part):
#
# solution = Program.to([[], delegated_puzzle_solution, []])
#
# So you do whatever you want with a bunch of coins at once and now you have two lists:
# 1) A list of G1Element objects yielded by AugSchemeMPL.sign
# 2) A list of CoinSpend objects.
#
# Now to spend them at once:
#
# signature = AugSchemeMPL.aggregate(signatures)
# spend_bundle = SpendBundle(coin_spends, signature)
#
async def combine_coins(self, coins: List[CoinWrapper]) -> Optional[SpendResult]:
# Overall structure:
# Create len-1 spends that just assert that the final coin is created with full value.
# Create 1 spend for the final coin that asserts the other spends occurred and
# Creates the new coin.
beginning_balance: uint64 = self.balance()
beginning_coins: int = len(self.usable_coins)
# We need the final coin to know what the announced coin name will be.
final_coin = CoinWrapper(
coins[-1].name(),
self.puzzle_hash,
uint64(sum(map(lambda x: x.amount, coins))),
self.puzzle,
)
destroyed_coin_spends: List[CoinSpend] = []
# Each coin wants agg_sig_me so we aggregate them at the end.
signatures: List[G2Element] = []
for c in coins[:-1]:
announce_conditions: List[List] = [
# Each coin expects the final coin creation announcement
[
ConditionOpcode.ASSERT_COIN_ANNOUNCEMENT,
std_hash(coins[-1].name() + final_coin.name()),
]
]
coin_spend, signature = c.create_standard_spend(self.sk_, announce_conditions)
destroyed_coin_spends.append(coin_spend)
signatures.append(signature)
final_coin_creation: List[List] = [
[ConditionOpcode.CREATE_COIN_ANNOUNCEMENT, final_coin.name()],
[ConditionOpcode.CREATE_COIN, self.puzzle_hash, final_coin.amount],
]
coin_spend, signature = coins[-1].create_standard_spend(self.sk_, final_coin_creation)
destroyed_coin_spends.append(coin_spend)
signatures.append(signature)
signature = AugSchemeMPL.aggregate(signatures)
spend_bundle = SpendBundle(destroyed_coin_spends, signature)
pushed: Dict[str, Union[str, List[Coin]]] = await self.parent.push_tx(spend_bundle)
# We should have the same amount of money.
assert beginning_balance == self.balance()
# We should have shredded n-1 coins and replaced one.
assert len(self.usable_coins) == beginning_coins - (len(coins) - 1)
return SpendResult(pushed)
# Find a coin containing amt we can use as a parent.
# Synthesize a coin with sufficient funds if possible.
async def choose_coin(self, amt) -> Optional[CoinWrapper]:
"""Given an amount requirement, find a coin that contains at least that much chia"""
start_balance: uint64 = self.balance()
coins_to_spend: Optional[List[Coin]] = self.compute_combine_action(amt, [], dict(self.usable_coins))
# Couldn't find a working combination.
if coins_to_spend is None:
return None
if len(coins_to_spend) == 1:
only_coin: Coin = coins_to_spend[0]
return CoinWrapper(
only_coin.parent_coin_info,
only_coin.puzzle_hash,
only_coin.amount,
self.puzzle,
)
# We receive a timeline of actions to take (indicating that we have a plan)
# Do the first action and start over.
result: Optional[SpendResult] = await self.combine_coins(
list(
map(
lambda x: CoinWrapper(x.parent_coin_info, x.puzzle_hash, x.amount, self.puzzle),
coins_to_spend,
)
)
)
if result is None:
return None
assert self.balance() == start_balance
return await self.choose_coin(amt)
# Create a new smart coin based on a parent coin and return the coin to the user.
# TODO:
# - allow use of more than one coin to launch smart coin
# - ensure input chia = output chia. it'd be dumb to just allow somebody
# to lose their chia without telling them.
async def launch_smart_coin(self, source: Program, **kwargs) -> Optional[CoinWrapper]:
"""Create a new smart coin based on a parent coin and return the smart coin's living
coin to the user or None if the spend failed."""
amt = uint64(1)
found_coin: Optional[CoinWrapper] = None
if "amt" in kwargs:
amt = kwargs["amt"]
if "launcher" in kwargs:
found_coin = kwargs["launcher"]
else:
found_coin = await self.choose_coin(amt)
if found_coin is None:
raise ValueError(f"could not find available coin containing {amt} mojo")
# Create a puzzle based on the incoming smart coin
cw = SmartCoinWrapper(DEFAULT_CONSTANTS.GENESIS_CHALLENGE, source)
condition_args: List[List] = [
[ConditionOpcode.CREATE_COIN, cw.puzzle_hash(), amt],
]
if amt < found_coin.amount:
condition_args.append([ConditionOpcode.CREATE_COIN, self.puzzle_hash, found_coin.amount - amt])
delegated_puzzle_solution = Program.to((1, condition_args))
solution = Program.to([[], delegated_puzzle_solution, []])
# Sign the (delegated_puzzle_hash + coin_name) with synthetic secret key
signature: G2Element = AugSchemeMPL.sign(
calculate_synthetic_secret_key(self.sk_, DEFAULT_HIDDEN_PUZZLE_HASH),
(
delegated_puzzle_solution.get_tree_hash()
+ found_coin.name()
+ DEFAULT_CONSTANTS.AGG_SIG_ME_ADDITIONAL_DATA
),
)
spend_bundle = SpendBundle(
[
CoinSpend(
found_coin.as_coin(), # Coin to spend
self.puzzle, # Puzzle used for found_coin
solution, # The solution to the puzzle locking found_coin
)
],
signature,
)
pushed: Dict[str, Union[str, List[Coin]]] = await self.parent.push_tx(spend_bundle)
if "error" not in pushed:
return cw.custom_coin(found_coin, amt)
else:
return None
# Give chia
async def give_chia(self, target: "Wallet", amt: uint64) -> Optional[CoinWrapper]:
return await self.launch_smart_coin(target.puzzle, amt=amt)
# Called each cycle before coins are re-established from the simulator.
def _clear_coins(self):
self.usable_coins = {}
# Public key of wallet
def pk(self) -> G1Element:
"""Return actor's public key"""
return self.pk_
# Balance of wallet
def balance(self) -> uint64:
"""Return the actor's balance in standard coins as we understand it"""
return uint64(sum(map(lambda x: x.amount, self.usable_coins.values())))
# Spend a coin, probably a smart coin.
# Allows the user to specify the arguments for the puzzle solution.
# Automatically takes care of signing, etc.
# Result is an object representing the actions taken when the block
# with this transaction was farmed.
async def spend_coin(self, coin: CoinWrapper, pushtx: bool = True, **kwargs) -> Union[SpendResult, SpendBundle]:
"""Given a coin object, invoke it on the blockchain, either as a standard
coin if no arguments are given or with custom arguments in args="""
amt = uint64(1)
if "amt" in kwargs:
amt = kwargs["amt"]
delegated_puzzle_solution: Optional[Program] = None
if "args" not in kwargs:
target_puzzle_hash: bytes32 = self.puzzle_hash
# Allow the user to 'give this much chia' to another user.
if "to" in kwargs:
target_puzzle_hash = kwargs["to"].puzzle_hash
# Automatic arguments from the user's intention.
if "custom_conditions" not in kwargs:
solution_list: List[List] = [[ConditionOpcode.CREATE_COIN, target_puzzle_hash, amt]]
else:
solution_list = kwargs["custom_conditions"]
if "remain" in kwargs:
remainer: Union[SmartCoinWrapper, Wallet] = kwargs["remain"]
remain_amt = uint64(coin.amount - amt)
if isinstance(remainer, SmartCoinWrapper):
solution_list.append(
[
ConditionOpcode.CREATE_COIN,
remainer.puzzle_hash(),
remain_amt,
]
)
elif isinstance(remainer, Wallet):
solution_list.append([ConditionOpcode.CREATE_COIN, remainer.puzzle_hash, remain_amt])
else:
raise ValueError("remainer is not a wallet or a smart coin")
delegated_puzzle_solution = Program.to((1, solution_list))
# Solution is the solution for the old coin.
solution = Program.to([[], delegated_puzzle_solution, []])
else:
delegated_puzzle_solution = Program.to(kwargs["args"])
solution = delegated_puzzle_solution
solution_for_coin = CoinSpend(
coin.as_coin(),
coin.puzzle(),
solution,
)
# The reason this use of sign_coin_spends exists is that it correctly handles
# the signing for non-standard coins. I don't fully understand the difference but
# this definitely does the right thing.
try:
spend_bundle: SpendBundle = await sign_coin_spends(
[solution_for_coin],
self.pk_to_sk,
DEFAULT_CONSTANTS.AGG_SIG_ME_ADDITIONAL_DATA,
DEFAULT_CONSTANTS.MAX_BLOCK_COST_CLVM,
)
except ValueError:
spend_bundle = SpendBundle(
[solution_for_coin],
G2Element(),
)
if pushtx:
pushed: Dict[str, Union[str, List[Coin]]] = await self.parent.push_tx(spend_bundle)
return SpendResult(pushed)
else:
return spend_bundle
# A user oriented (domain specific) view of the chia network.
class Network:
"""An object that owns a simulation, responsible for managing Wallet actors,
time and initialization."""
time: datetime.timedelta
sim: SpendSim
sim_client: SimClient
wallets: Dict[str, Wallet]
nobody: Wallet
@classmethod
async def create(cls) -> "Network":
self = cls()
self.time = datetime.timedelta(days=18750, seconds=61201) # Past the initial transaction freeze
self.sim = await SpendSim.create()
self.sim_client = SimClient(self.sim)
self.wallets = {}
self.nobody = self.make_wallet("nobody")
self.wallets[str(self.nobody.pk())] = self.nobody
return self
async def close(self):
await self.sim.close()
# Have the system farm one block with a specific beneficiary (nobody if not specified).
async def farm_block(self, **kwargs) -> Tuple[List[Coin], List[Coin]]:
"""Given a farmer, farm a block with that actor as the beneficiary of the farm
reward.
Used for causing chia balance to exist so the system can do things.
"""
farmer: Wallet = self.nobody
if "farmer" in kwargs:
farmer = kwargs["farmer"]
farm_duration = datetime.timedelta(block_time)
farmed: Tuple[List[Coin], List[Coin]] = await self.sim.farm_block(farmer.puzzle_hash)
for k, w in self.wallets.items():
w._clear_coins()
for kw, w in self.wallets.items():
coin_records: List[CoinRecord] = await self.sim_client.get_coin_records_by_puzzle_hash(w.puzzle_hash)
for coin_record in coin_records:
if coin_record.spent is False:
w.add_coin(CoinWrapper.from_coin(coin_record.coin, w.puzzle))
self.time += farm_duration
return farmed
def _alloc_key(self) -> Tuple[G1Element, PrivateKey]:
key_idx: int = len(self.wallets)
pk: G1Element = public_key_for_index(key_idx)
priv: PrivateKey = private_key_for_index(key_idx)
return pk, priv
# Allow the user to create a wallet identity to whom standard coins may be targeted.
# This results in the creation of a wallet that tracks balance and standard coins.
# Public and private key from here are used in signing.
def make_wallet(self, name: str) -> Wallet:
"""Create a wallet for an actor. This causes the actor's chia balance in standard
coin to be tracked during the simulation. Wallets have some domain specific methods
that behave in similar ways to other blockchains."""
pk, priv = self._alloc_key()
w = Wallet(self, name, pk, priv)
self.wallets[str(w.pk())] = w
return w
# Skip real time by farming blocks until the target duration is achieved.
async def skip_time(self, target_duration: str, **kwargs):
"""Skip a duration of simulated time, causing blocks to be farmed. If a farmer
is specified, they win each block"""
target_time = self.time + datetime.timedelta(pytimeparse.parse(target_duration) / duration_div)
while target_time > self.get_timestamp():
await self.farm_block(**kwargs)
self.sim.pass_time(uint64(20))
# Or possibly aggregate farm_block results.
return None
def get_timestamp(self) -> datetime.timedelta:
"""Return the current simualtion time in seconds."""
return datetime.timedelta(seconds=self.sim.timestamp)
# Given a spend bundle, farm a block and analyze the result.
async def push_tx(self, bundle: SpendBundle) -> Dict[str, Union[str, List[Coin]]]:
"""Given a spend bundle, try to farm a block containing it. If the spend bundle
didn't validate, then a result containing an 'error' key is returned. The reward
for the block goes to Network::nobody"""
status, error = await self.sim_client.push_tx(bundle)
if error:
return {"error": str(error)}
# Common case that we want to farm this right away.
additions, removals = await self.farm_block()
return {
"additions": additions,
"removals": removals,
}
async def setup_oracle() -> Tuple[Network, Wallet, Wallet, Wallet]:
network: Network = await Network.create()
oracle: Wallet = network.make_wallet("oracle")
alice: Wallet = network.make_wallet("alice")
bob: Wallet = network.make_wallet("bob")
await network.farm_block()
await network.farm_block(farmer=oracle)
return network, oracle, alice, bob
async def setup_two_wallet_node() -> Tuple[Network, Wallet, Wallet]:
network: Network = await Network.create()
alice: Wallet = network.make_wallet("alice")
bob: Wallet = network.make_wallet("bob")
await network.farm_block()
await network.farm_block(farmer=alice)
await network.farm_block(farmer=bob)
return network, alice, bob
async def setup_node_only():
node = await Network.create()
await node.farm_block()
return node
def load_clsp_relative(filename: str, search_paths: List[Path] = [Path("include/")]):
base = Path().parent.resolve()
source = base / filename
target = base / f"{filename}.hex"
searches = [base / s for s in search_paths]
compile_clvm(source, target, searches)
clvm = target.read_text()
clvm_blob = bytes.fromhex(clvm)
sp = SerializedProgram.from_bytes(clvm_blob)
return Program.from_bytes(bytes(sp))