1
#include "emulator-extern.h"
2
#include "td/utils/logging.h"
3
#include "td/utils/JsonBuilder.h"
4
#include "td/utils/misc.h"
5
#include "td/utils/optional.h"
8
#include "crypto/common/bitstring.h"
10
struct TransactionEmulationParams {
13
td::optional<std::string> rand_seed_hex;
20
td::Result<TransactionEmulationParams> decode_transaction_emulation_params(const char* json) {
21
TransactionEmulationParams params;
23
std::string json_str(json);
24
TRY_RESULT(input_json, td::json_decode(td::MutableSlice(json_str)));
25
auto &obj = input_json.get_object();
27
TRY_RESULT(utime_field, td::get_json_object_field(obj, "utime", td::JsonValue::Type::Number, false));
28
TRY_RESULT(utime, td::to_integer_safe<td::uint32>(utime_field.get_number()));
31
TRY_RESULT(lt_field, td::get_json_object_field(obj, "lt", td::JsonValue::Type::String, false));
32
TRY_RESULT(lt, td::to_integer_safe<td::uint64>(lt_field.get_string()));
35
TRY_RESULT(rand_seed_str, td::get_json_object_string_field(obj, "rand_seed", true));
36
if (rand_seed_str.size() > 0) {
37
params.rand_seed_hex = rand_seed_str;
40
TRY_RESULT(ignore_chksig, td::get_json_object_bool_field(obj, "ignore_chksig", false));
41
params.ignore_chksig = ignore_chksig;
43
TRY_RESULT(debug_enabled, td::get_json_object_bool_field(obj, "debug_enabled", false));
44
params.debug_enabled = debug_enabled;
46
TRY_RESULT(is_tick_tock, td::get_json_object_bool_field(obj, "is_tick_tock", true, false));
47
params.is_tick_tock = is_tick_tock;
49
TRY_RESULT(is_tock, td::get_json_object_bool_field(obj, "is_tock", true, false));
50
params.is_tock = is_tock;
52
if (is_tock && !is_tick_tock) {
53
return td::Status::Error("Inconsistent parameters is_tick_tock=false, is_tock=true");
59
struct GetMethodParams {
63
td::optional<std::string> libs;
64
td::optional<std::string> prev_blocks_info;
68
std::string rand_seed_hex;
74
td::Result<GetMethodParams> decode_get_method_params(const char* json) {
75
GetMethodParams params;
77
std::string json_str(json);
78
TRY_RESULT(input_json, td::json_decode(td::MutableSlice(json_str)));
79
auto &obj = input_json.get_object();
81
TRY_RESULT(code, td::get_json_object_string_field(obj, "code", false));
84
TRY_RESULT(data, td::get_json_object_string_field(obj, "data", false));
87
TRY_RESULT(verbosity, td::get_json_object_int_field(obj, "verbosity", false));
88
params.verbosity = verbosity;
90
TRY_RESULT(libs, td::get_json_object_string_field(obj, "libs", true));
91
if (libs.size() > 0) {
95
TRY_RESULT(prev_blocks_info, td::get_json_object_string_field(obj, "prev_blocks_info", true));
96
if (prev_blocks_info.size() > 0) {
97
params.prev_blocks_info = prev_blocks_info;
100
TRY_RESULT(address, td::get_json_object_string_field(obj, "address", false));
101
params.address = address;
103
TRY_RESULT(unixtime_field, td::get_json_object_field(obj, "unixtime", td::JsonValue::Type::Number, false));
104
TRY_RESULT(unixtime, td::to_integer_safe<td::uint32>(unixtime_field.get_number()));
105
params.unixtime = unixtime;
107
TRY_RESULT(balance_field, td::get_json_object_field(obj, "balance", td::JsonValue::Type::String, false));
108
TRY_RESULT(balance, td::to_integer_safe<td::uint64>(balance_field.get_string()));
109
params.balance = balance;
111
TRY_RESULT(rand_seed_str, td::get_json_object_string_field(obj, "rand_seed", false));
112
params.rand_seed_hex = rand_seed_str;
114
TRY_RESULT(gas_limit_field, td::get_json_object_field(obj, "gas_limit", td::JsonValue::Type::String, false));
115
TRY_RESULT(gas_limit, td::to_integer_safe<td::uint64>(gas_limit_field.get_string()));
116
params.gas_limit = gas_limit;
118
TRY_RESULT(method_id, td::get_json_object_int_field(obj, "method_id", false));
119
params.method_id = method_id;
121
TRY_RESULT(debug_enabled, td::get_json_object_bool_field(obj, "debug_enabled", false));
122
params.debug_enabled = debug_enabled;
127
class NoopLog : public td::LogInterface {
132
void append(td::CSlice new_slice, int log_level) override {
135
void rotate() override {
141
void* create_emulator(const char *config, int verbosity) {
144
td::log_interface = &logger;
146
SET_VERBOSITY_LEVEL(verbosity_NEVER);
147
return transaction_emulator_create(config, verbosity);
150
void destroy_emulator(void* em) {
153
td::log_interface = &logger;
155
SET_VERBOSITY_LEVEL(verbosity_NEVER);
156
transaction_emulator_destroy(em);
159
const char *emulate_with_emulator(void* em, const char* libs, const char* account, const char* message, const char* params) {
162
td::log_interface = &logger;
163
SET_VERBOSITY_LEVEL(verbosity_DEBUG);
165
auto decoded_params_res = decode_transaction_emulation_params(params);
166
if (decoded_params_res.is_error()) {
167
return strdup(R"({"fail":true,"message":"Can't decode other params"})");
169
auto decoded_params = decoded_params_res.move_as_ok();
171
bool rand_seed_set = true;
172
if (decoded_params.rand_seed_hex) {
173
rand_seed_set = transaction_emulator_set_rand_seed(em, decoded_params.rand_seed_hex.unwrap().c_str());
176
if (!transaction_emulator_set_libs(em, libs) ||
177
!transaction_emulator_set_lt(em, decoded_params.lt) ||
178
!transaction_emulator_set_unixtime(em, decoded_params.utime) ||
179
!transaction_emulator_set_ignore_chksig(em, decoded_params.ignore_chksig) ||
180
!transaction_emulator_set_debug_enabled(em, decoded_params.debug_enabled) ||
182
transaction_emulator_destroy(em);
183
return strdup(R"({"fail":true,"message":"Can't set params"})");
187
if (decoded_params.is_tick_tock) {
188
result = transaction_emulator_emulate_tick_tock_transaction(em, account, decoded_params.is_tock);
190
result = transaction_emulator_emulate_transaction(em, account, message);
193
const char* output = nullptr;
196
auto json_obj = jb.enter_object();
197
json_obj("output", td::JsonRaw(td::Slice(result)));
198
json_obj("logs", logger.get_string());
200
output = strdup(jb.string_builder().as_cslice().c_str());
202
free((void*) result);
207
const char *emulate(const char *config, const char* libs, int verbosity, const char* account, const char* message, const char* params) {
208
auto em = transaction_emulator_create(config, verbosity);
209
auto result = emulate_with_emulator(em, libs, account, message, params);
210
transaction_emulator_destroy(em);
214
const char *run_get_method(const char *params, const char* stack, const char* config) {
217
td::log_interface = &logger;
218
SET_VERBOSITY_LEVEL(verbosity_DEBUG);
220
auto decoded_params_res = decode_get_method_params(params);
221
if (decoded_params_res.is_error()) {
222
return strdup(R"({"fail":true,"message":"Can't decode params"})");
224
auto decoded_params = decoded_params_res.move_as_ok();
226
auto tvm = tvm_emulator_create(decoded_params.code.c_str(), decoded_params.data.c_str(), decoded_params.verbosity);
228
if ((decoded_params.libs && !tvm_emulator_set_libraries(tvm, decoded_params.libs.value().c_str())) ||
229
!tvm_emulator_set_c7(tvm, decoded_params.address.c_str(), decoded_params.unixtime, decoded_params.balance,
230
decoded_params.rand_seed_hex.c_str(), config) ||
231
(decoded_params.prev_blocks_info &&
232
!tvm_emulator_set_prev_blocks_info(tvm, decoded_params.prev_blocks_info.value().c_str())) ||
233
(decoded_params.gas_limit > 0 && !tvm_emulator_set_gas_limit(tvm, decoded_params.gas_limit)) ||
234
!tvm_emulator_set_debug_enabled(tvm, decoded_params.debug_enabled)) {
235
tvm_emulator_destroy(tvm);
236
return strdup(R"({"fail":true,"message":"Can't set params"})");
239
auto res = tvm_emulator_run_get_method(tvm, decoded_params.method_id, stack);
241
tvm_emulator_destroy(tvm);
243
const char* output = nullptr;
246
auto json_obj = jb.enter_object();
247
json_obj("output", td::JsonRaw(td::Slice(res)));
248
json_obj("logs", logger.get_string());
250
output = strdup(jb.string_builder().as_cslice().c_str());