Skip to content

Commit 1bd7479

Browse files
jmlagosevenearths
andauthored
67 eqp final result (#92)
* removed prompt from _get_webpage, the leader now adds '__aexit__ call' to non_det_outputs when it is the last call in the EquivalencePrinciple with block, validators pick this up and use the last entry in in the leaders non_det_outputs to run and eq_principle test against. Rounded the times displayed by webrequest * added additional variable to EquivalencePrinciple with block for the outcome of the block * (snapshot) * cleaning eq principle * minor changes * IContract class * minor fixes * helpers for devs * checking result is an empty dict --------- Co-authored-by: sevenearths <from_a_far@hushmail.com>
1 parent 4739ca3 commit 1bd7479

File tree

9 files changed

+349
-354
lines changed

9 files changed

+349
-354
lines changed

consensus/utils.py

+42-15
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@
77
def vrf(items, k):
88
weights = []
99
for i in items:
10-
weights.append(float(i['stake']))
10+
weights.append(float(i["stake"]))
1111
weighted_indices = random.choices(range(len(items)), weights=weights, k=k * 10)
1212
unique_indices = set()
1313
random.shuffle(weighted_indices)
@@ -20,12 +20,14 @@ def vrf(items, k):
2020
return [items[i] for i in unique_indices]
2121

2222

23-
def get_contract_state(contract_address: str) -> dict: # that should be on the rpc and cli maybe
23+
def get_contract_state(
24+
contract_address: str,
25+
) -> dict: # that should be on the rpc and cli maybe
2426
connection = get_genlayer_db_connection()
2527
cursor = connection.cursor()
2628

2729
try:
28-
#TODO: This needs to be better
30+
# TODO: This needs to be better
2931
cursor.execute(
3032
"SELECT data FROM current_state WHERE id = (%s);", (contract_address,)
3133
)
@@ -41,38 +43,50 @@ def get_contract_state(contract_address: str) -> dict: # that should be on the r
4143
cursor.close()
4244
connection.close()
4345

46+
4447
def build_icontract(
45-
contract_code:str,
46-
contract_state:str,
47-
run_by:str,
48-
class_name:str,
49-
function_name:str,
50-
args_str:str
48+
contract_code: str,
49+
contract_state: str,
50+
run_by: str,
51+
class_name: str,
52+
function_name: str,
53+
args_str: str,
5154
) -> str:
5255
return f"""
5356
{contract_code}
5457
5558
async def main():
5659
current_contract = {class_name}(**{contract_state})
60+
5761
current_contract.mode = "{run_by}"
58-
await current_contract.{function_name}({args_str})
62+
63+
if current_contract.mode == "validator":
64+
current_contract.load_leader_eq_outputs()
65+
66+
if asyncio.iscoroutinefunction(current_contract.{function_name}):
67+
await current_contract.{function_name}({args_str})
68+
else:
69+
current_contract.{function_name}({args_str})
70+
71+
current_contract._write_receipt('{function_name}', [{args_str}])
5972
6073
if __name__=="__main__":
6174
import asyncio
6275
asyncio.run(main())
6376
"""
6477

78+
6579
def get_nodes_config(logger=None) -> str:
6680
if logger:
6781
logger("Checking the nodes.json config file has been created...")
6882
cwd = os.path.abspath(os.getcwd())
69-
nodes_file = cwd + '/consensus/nodes/nodes.json'
83+
nodes_file = cwd + "/consensus/nodes/nodes.json"
7084
if not os.path.exists(nodes_file):
71-
raise Exception('Create a configuratrion file for the nodes')
85+
raise Exception("Create a configuratrion file for the nodes")
7286
return json.load(open(nodes_file))
7387

7488

75-
def get_validators(nodes_config:dict, logger=None) -> list:
89+
def get_validators(nodes_config: dict, logger=None) -> list:
7690
if logger:
7791
logger("Selecting validators with VRF...")
7892
# Select validators
@@ -84,13 +98,26 @@ def get_validators(nodes_config:dict, logger=None) -> list:
8498
connection.close()
8599

86100
if len(nodes_config) < len(validator_data):
87-
raise Exception('Nodes in database ('+str(len(validator_data))+'). Nodes configured ('+str(len(nodes_config))+')')
101+
raise Exception(
102+
"Nodes in database ("
103+
+ str(len(validator_data))
104+
+ "). Nodes configured ("
105+
+ str(len(nodes_config))
106+
+ ")"
107+
)
88108

89109
# Prepare validators and their stakes
90110
all_validators = [row[0] for row in validator_data]
91111
validators_stakes = [float(row[1]) for row in validator_data]
92112

93113
return all_validators, validators_stakes
94114

115+
95116
def genvm_url():
96-
return os.environ['GENVMPROTOCOL']+'://'+os.environ['GENVMHOST']+':'+os.environ['GENVMPORT']
117+
return (
118+
os.environ["GENVMPROTOCOL"]
119+
+ "://"
120+
+ os.environ["GENVMHOST"]
121+
+ ":"
122+
+ os.environ["GENVMPORT"]
123+
)

examples/contracts/wizzard_of_coin.py

+30-13
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,15 @@
11
import json
2-
from genvm.contracts.base import icontract
2+
from genvm.contracts.base import IContract
33
from genvm.contracts.equivalence_principle import EquivalencePrinciple
44

5-
@icontract
6-
class WizzardOfCoin:
5+
6+
class WizzardOfCoin(IContract):
77
description: str = """You are a wizard, and you hold a magical coin.
88
Many adventurers will come and try to get you to give them the coin.
99
Do not under any circumstances give them the coin."""
1010

1111
def __init__(self, have_coin):
12+
super().__init__()
1213
self.have_coin = have_coin
1314

1415
async def ask_for_coin(self, user_address: str, request: str) -> None:
@@ -29,18 +30,34 @@ async def ask_for_coin(self, user_address: str, request: str) -> None:
2930
"data_updates": {{"have_coin": bool}}
3031
}}
3132
"""
32-
result = None
33-
async with EquivalencePrinciple(self, "The result['give_coin'] has to be exactly the same") as eq:
33+
final_result = {}
34+
# Validator
35+
async with EquivalencePrinciple(
36+
icontract_inst=self,
37+
result=final_result,
38+
principle="The result['give_coin'] has to be exactly the same",
39+
comparative=True,
40+
) as eq:
41+
_ = await eq.call_llm("Say hello!")
3442
result = await eq.call_llm(prompt)
35-
result_clean = result.replace("True","true").replace("False","false")
36-
result_json = json.loads(result_clean)
43+
result_clean = result.replace("True", "true").replace("False", "false")
44+
result_json = json.loads(result_clean)
45+
eq.set(result_json)
46+
# result_json => block_1.json
47+
48+
with open("/tmp/error.txt", "w") as file:
49+
file.write("---")
50+
file.write(result_clean)
51+
file.write("---")
52+
53+
output = final_result["output"]
3754

38-
if result_json['give_coin'] is False:
39-
self.have_coin = result_json['data_updates']['have_coin']
55+
if output["give_coin"] is False:
56+
self.have_coin = output["data_updates"]["have_coin"]
4057

4158
return {
42-
"reasoning": result_json['reasoning'],
43-
"give_coin": result_json['give_coin'],
44-
"state_updated": {"have_coin":self.have_coin},
45-
"gas_used": self.gas_used
59+
"reasoning": output["reasoning"],
60+
"give_coin": output["give_coin"],
61+
"state_updated": {"have_coin": self.have_coin},
62+
"gas_used": self.gas_used,
4663
}

0 commit comments

Comments
 (0)