forked from starknet-edu/starknet-cairo-101
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathex12.cairo
140 lines (125 loc) · 5.38 KB
/
ex12.cairo
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
// ######## Ex 12
// Events
// In this exercise, you need to:
// - Use a function to get assigned a private variable
// - Use a function to emit event with the value of the private variable
// - Your points are credited by the contract
#[contract]
mod Ex12 {
////////////////////////////////
// Core Library imports
// These are syscalls and functionalities that allow you to write Starknet contracts
////////////////////////////////
use starknet::get_caller_address;
use starknet::ContractAddress;
use array::ArrayTrait;
use option::OptionTrait;
////////////////////////////////
// Internal imports
// These functions become part of the set of functions of the contract
////////////////////////////////
use starknet_cairo_101::utils::ex00_base::Ex00Base::distribute_points;
use starknet_cairo_101::utils::ex00_base::Ex00Base::validate_exercise;
use starknet_cairo_101::utils::ex00_base::Ex00Base::ex_initializer;
use starknet_cairo_101::utils::helper;
use starknet_cairo_101::utils::ex00_base::Ex00Base::update_class_hash_by_admin;
////////////////////////////////
// Storage
// In Cairo 1, storage is declared in a struct
// Storage is not visible by default through the ABI
////////////////////////////////
struct Storage {
user_slots: LegacyMap::<ContractAddress, u128>,
values_mapped_secret: LegacyMap::<u128, u128>,
was_initialized: bool,
next_slot: u128,
}
////////////////////////////////
// EVENTS
////////////////////////////////
#[event]
fn Assign_User_Slot_Called(account: ContractAddress, secret_value: u128) {}
////////////////////////////////
// Constructor
// This function (indicated with #[constructor]) is called when the contract is deployed and is used to initialize the contract's state
////////////////////////////////
#[constructor]
fn constructor(
_tderc20_address: ContractAddress, _players_registry: ContractAddress, _workshop_id: u128, _exercise_id: u128
) {
ex_initializer(_tderc20_address, _players_registry, _workshop_id, _exercise_id);
}
////////////////////////////////
// View Functions
// Public variables should be declared explicitly with a getter function (indicated with #[view]) to be visible through the ABI and callable from other contracts
////////////////////////////////
#[view]
fn get_user_slots(account: ContractAddress) -> u128 {
return user_slots::read(account);
}
////////////////////////////////
// External functions
// These functions are callable by other contracts or external calls such as DAPP, which are indicated with #[external] (similar to "public" in Solidity)
////////////////////////////////
#[external]
fn claim_points(expected_value: u128) {
// Reading caller address
let sender_address: ContractAddress = get_caller_address();
// Checking that the user got a slot assigned
let user_slot = user_slots::read(sender_address);
assert(user_slot != 0_u128, 'ASSIGN_USER_SLOT_FIRST');
// Checking that the value provided by the user is the one we expect
// Still sneaky.
// Or not. Is this psyops?
let value = values_mapped_secret::read(user_slot);
assert(value == expected_value, 'NOT_EXPECTED_SECRET_VALUE');
// Checking if the user has validated the exercise before
validate_exercise(sender_address);
// Sending points to the address specified as parameter
distribute_points(sender_address, 2_u128);
}
#[external]
fn assign_user_slot() {
// Reading caller address
let sender_address: ContractAddress = get_caller_address();
let next_slot_temp = next_slot::read();
let next_value = values_mapped_secret::read(next_slot_temp + 1_u128);
if next_value == 0_u128 {
user_slots::write(sender_address, 1_u128);
next_slot::write(0_u128);
} else {
user_slots::write(sender_address, next_slot_temp + 1_u128);
next_slot::write(next_slot_temp + 1_u128);
}
let user_slot = user_slots::read(sender_address);
let secret_value = values_mapped_secret::read(user_slot);
// Emit an event with secret value
Assign_User_Slot_Called(sender_address, secret_value + 32_u128);
}
////////////////////////////////
// External functions - Administration
// Only admins can call these. You don't need to understand them to finish the exercise.
////////////////////////////////
#[external]
fn update_class_hash(class_hash: felt252) {
update_class_hash_by_admin(class_hash);
}
#[external]
fn set_random_values(values: Array::<u128>) {
// Check if the random values were already initialized
let was_initialized_read = was_initialized::read();
assert(was_initialized_read != true, 'NOT_INITIALISED');
let mut idx: u128 = 0_u128;
set_a_random_value(idx, values);
// Mark that value store was initialized
was_initialized::write(true);
}
fn set_a_random_value(mut idx: u128, mut values: Array::<u128>) {
helper::check_gas();
if !values.is_empty() {
values_mapped_secret::write(idx, values.pop_front().unwrap());
idx = idx + 1_u128;
set_a_random_value(idx, values);
}
}
}