2026-02-03 18:12:53 +01:00
|
|
|
#include "logos_execution_zone_wallet_module.h"
|
|
|
|
|
|
|
|
|
|
#include <QtCore/QDebug>
|
2026-02-18 21:23:16 +01:00
|
|
|
#include <QtCore/QJsonArray>
|
2026-02-18 22:10:38 +01:00
|
|
|
#include <QtCore/QJsonDocument>
|
|
|
|
|
#include <QtCore/QJsonObject>
|
2026-02-18 21:23:16 +01:00
|
|
|
#include <QtCore/QVariantMap>
|
|
|
|
|
|
|
|
|
|
static QString bytesToHex(const uint8_t* data, const size_t length) {
|
|
|
|
|
const QByteArray bytearray(reinterpret_cast<const char*>(data), static_cast<int>(length));
|
|
|
|
|
return QString::fromLatin1(bytearray.toHex());
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-18 22:10:38 +01:00
|
|
|
namespace JsonKeys {
|
2026-02-18 22:52:28 +01:00
|
|
|
static constexpr auto TxHash = "tx_hash";
|
|
|
|
|
static constexpr auto Success = "success";
|
|
|
|
|
static constexpr auto ProgramOwner = "program_owner";
|
|
|
|
|
static constexpr auto Balance = "balance";
|
|
|
|
|
static constexpr auto Nonce = "nonce";
|
|
|
|
|
static constexpr auto Data = "data";
|
|
|
|
|
static constexpr auto NullifierPublicKey = "nullifier_public_key";
|
|
|
|
|
static constexpr auto ViewingPublicKey = "viewing_public_key";
|
|
|
|
|
static constexpr auto AccountId = "account_id";
|
|
|
|
|
static constexpr auto IsPublic = "is_public";
|
|
|
|
|
} // namespace JsonKeys
|
2026-02-18 22:10:38 +01:00
|
|
|
|
2026-02-18 21:23:16 +01:00
|
|
|
static bool hexToBytes(const QString& hex, QByteArray& output_bytes, int expectedLength = -1) {
|
|
|
|
|
QString trimmed_hex = hex.trimmed();
|
2026-02-18 22:52:28 +01:00
|
|
|
if (trimmed_hex.startsWith("0x", Qt::CaseInsensitive))
|
|
|
|
|
trimmed_hex = trimmed_hex.mid(2);
|
|
|
|
|
if (trimmed_hex.size() % 2 != 0)
|
|
|
|
|
return false;
|
2026-02-18 21:23:16 +01:00
|
|
|
const QByteArray decoded = QByteArray::fromHex(trimmed_hex.toLatin1());
|
2026-02-18 22:52:28 +01:00
|
|
|
if (expectedLength != -1 && decoded.size() != expectedLength)
|
|
|
|
|
return false;
|
2026-02-18 21:23:16 +01:00
|
|
|
output_bytes = decoded;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static bool hexToU128(const QString& hex, uint8_t (*output)[16]) {
|
|
|
|
|
QByteArray buffer;
|
2026-02-18 22:52:28 +01:00
|
|
|
if (!hexToBytes(hex, buffer, 16))
|
|
|
|
|
return false;
|
2026-02-18 21:23:16 +01:00
|
|
|
memcpy(output, buffer.constData(), 16);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static QString bytes32ToHex(const FfiBytes32& bytes) {
|
|
|
|
|
return bytesToHex(bytes.data, 32);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static bool hexToBytes32(const QString& hex, FfiBytes32* output_bytes) {
|
2026-02-18 22:52:28 +01:00
|
|
|
if (output_bytes == nullptr)
|
|
|
|
|
return false;
|
2026-02-18 21:23:16 +01:00
|
|
|
QByteArray buffer;
|
2026-02-18 22:52:28 +01:00
|
|
|
if (!hexToBytes(hex, buffer, 32))
|
|
|
|
|
return false;
|
2026-02-18 21:23:16 +01:00
|
|
|
memcpy(output_bytes->data, buffer.constData(), 32);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2026-02-03 18:12:53 +01:00
|
|
|
|
2026-02-18 22:10:38 +01:00
|
|
|
static QString ffiTransferResultToJson(const FfiTransferResult& result) {
|
|
|
|
|
QVariantMap map;
|
|
|
|
|
map[JsonKeys::TxHash] = result.tx_hash ? QString::fromUtf8(result.tx_hash) : QString();
|
|
|
|
|
map[JsonKeys::Success] = result.success;
|
|
|
|
|
return QJsonDocument(QJsonObject::fromVariantMap(map)).toJson(QJsonDocument::Compact);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static QString ffiAccountToJson(const FfiAccount& account) {
|
|
|
|
|
QVariantMap map;
|
|
|
|
|
map[JsonKeys::ProgramOwner] = bytesToHex(reinterpret_cast<const uint8_t*>(account.program_owner.data), 32);
|
|
|
|
|
map[JsonKeys::Balance] = bytesToHex(account.balance.data, 16);
|
|
|
|
|
map[JsonKeys::Nonce] = bytesToHex(account.nonce.data, 16);
|
|
|
|
|
if (account.data && account.data_len > 0) {
|
|
|
|
|
map[JsonKeys::Data] = bytesToHex(account.data, account.data_len);
|
|
|
|
|
} else {
|
|
|
|
|
map[JsonKeys::Data] = QString();
|
|
|
|
|
}
|
|
|
|
|
return QJsonDocument(QJsonObject::fromVariantMap(map)).toJson(QJsonDocument::Compact);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static QJsonObject ffiAccountListEntryToJson(const FfiAccountListEntry& entry) {
|
|
|
|
|
QVariantMap map;
|
|
|
|
|
map[JsonKeys::AccountId] = bytes32ToHex(entry.account_id);
|
|
|
|
|
map[JsonKeys::IsPublic] = entry.is_public;
|
|
|
|
|
return QJsonObject::fromVariantMap(map);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static QString ffiPrivateAccountKeysToJson(const FfiPrivateAccountKeys& keys) {
|
|
|
|
|
QVariantMap map;
|
|
|
|
|
map[JsonKeys::NullifierPublicKey] = bytes32ToHex(keys.nullifier_public_key);
|
|
|
|
|
if (keys.viewing_public_key && keys.viewing_public_key_len > 0) {
|
|
|
|
|
map[JsonKeys::ViewingPublicKey] = bytesToHex(keys.viewing_public_key, keys.viewing_public_key_len);
|
|
|
|
|
} else {
|
|
|
|
|
map[JsonKeys::ViewingPublicKey] = QString();
|
|
|
|
|
}
|
|
|
|
|
return QJsonDocument(QJsonObject::fromVariantMap(map)).toJson(QJsonDocument::Compact);
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-18 22:18:10 +01:00
|
|
|
static bool jsonToFfiPrivateAccountKeys(const QString& json, FfiPrivateAccountKeys* output_keys) {
|
2026-02-18 22:10:38 +01:00
|
|
|
QJsonDocument doc = QJsonDocument::fromJson(json.toUtf8());
|
2026-02-18 22:52:28 +01:00
|
|
|
if (!doc.isObject())
|
|
|
|
|
return false;
|
2026-02-18 22:10:38 +01:00
|
|
|
const QVariantMap map = doc.object().toVariantMap();
|
2026-02-18 22:52:28 +01:00
|
|
|
|
2026-02-18 22:10:38 +01:00
|
|
|
if (map.contains(JsonKeys::NullifierPublicKey)) {
|
2026-02-18 22:52:28 +01:00
|
|
|
if (!hexToBytes32(map[JsonKeys::NullifierPublicKey].toString(), &output_keys->nullifier_public_key))
|
|
|
|
|
return false;
|
2026-02-18 22:10:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (map.contains(JsonKeys::ViewingPublicKey)) {
|
|
|
|
|
QByteArray buffer;
|
2026-02-18 22:52:28 +01:00
|
|
|
if (!hexToBytes(map[JsonKeys::ViewingPublicKey].toString(), buffer))
|
|
|
|
|
return false;
|
|
|
|
|
|
2026-02-18 22:10:38 +01:00
|
|
|
uint8_t* data = static_cast<uint8_t*>(malloc(buffer.size()));
|
|
|
|
|
memcpy(data, buffer.constData(), buffer.size());
|
2026-02-18 22:18:10 +01:00
|
|
|
output_keys->viewing_public_key = data;
|
|
|
|
|
output_keys->viewing_public_key_len = buffer.size();
|
2026-02-18 22:10:38 +01:00
|
|
|
} else {
|
2026-02-18 22:18:10 +01:00
|
|
|
output_keys->viewing_public_key = nullptr;
|
|
|
|
|
output_keys->viewing_public_key_len = 0;
|
2026-02-18 22:10:38 +01:00
|
|
|
}
|
2026-02-18 22:52:28 +01:00
|
|
|
|
2026-02-18 22:10:38 +01:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-04 16:18:56 +01:00
|
|
|
LogosExecutionZoneWalletModule::LogosExecutionZoneWalletModule() = default;
|
2026-02-18 11:21:00 +00:00
|
|
|
|
|
|
|
|
LogosExecutionZoneWalletModule::~LogosExecutionZoneWalletModule() {
|
|
|
|
|
if (walletHandle) {
|
|
|
|
|
wallet_ffi_destroy(walletHandle);
|
|
|
|
|
walletHandle = nullptr;
|
|
|
|
|
}
|
|
|
|
|
}
|
2026-02-03 18:12:53 +01:00
|
|
|
|
2026-02-04 15:51:02 +01:00
|
|
|
// === Plugin Interface ===
|
2026-02-03 18:12:53 +01:00
|
|
|
|
|
|
|
|
QString LogosExecutionZoneWalletModule::name() const {
|
2026-02-20 11:37:05 +01:00
|
|
|
return "liblogos_execution_zone_wallet_module";
|
2026-02-03 18:12:53 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString LogosExecutionZoneWalletModule::version() const {
|
|
|
|
|
return "1.0.0";
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-04 15:51:02 +01:00
|
|
|
// === Logos Core ===
|
|
|
|
|
|
2026-02-04 16:18:56 +01:00
|
|
|
void LogosExecutionZoneWalletModule::initLogos(LogosAPI* logosApiInstance) {
|
2026-02-20 11:37:05 +01:00
|
|
|
logosAPI = logosApiInstance;
|
2026-02-04 15:51:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// === Account Management ===
|
|
|
|
|
|
2026-02-20 08:59:18 +00:00
|
|
|
QString LogosExecutionZoneWalletModule::create_account_public() {
|
2026-02-18 21:23:16 +01:00
|
|
|
FfiBytes32 id{};
|
|
|
|
|
const WalletFfiError error = wallet_ffi_create_account_public(walletHandle, &id);
|
2026-02-20 08:59:18 +00:00
|
|
|
if (error != SUCCESS) {
|
|
|
|
|
qWarning() << "create_account_public: wallet FFI error" << error;
|
|
|
|
|
return {};
|
2026-02-18 21:23:16 +01:00
|
|
|
}
|
2026-02-20 08:59:18 +00:00
|
|
|
return bytes32ToHex(id);
|
2026-02-04 15:51:02 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-20 08:59:18 +00:00
|
|
|
QString LogosExecutionZoneWalletModule::create_account_private() {
|
2026-02-18 21:23:16 +01:00
|
|
|
FfiBytes32 id{};
|
|
|
|
|
const WalletFfiError error = wallet_ffi_create_account_private(walletHandle, &id);
|
2026-02-20 08:59:18 +00:00
|
|
|
if (error != SUCCESS) {
|
|
|
|
|
qWarning() << "create_account_private: wallet FFI error" << error;
|
|
|
|
|
return {};
|
2026-02-18 21:23:16 +01:00
|
|
|
}
|
2026-02-20 08:59:18 +00:00
|
|
|
return bytes32ToHex(id);
|
2026-02-04 15:51:02 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-20 08:59:18 +00:00
|
|
|
QJsonArray LogosExecutionZoneWalletModule::list_accounts() {
|
2026-02-18 22:10:38 +01:00
|
|
|
FfiAccountList list{};
|
|
|
|
|
const WalletFfiError error = wallet_ffi_list_accounts(walletHandle, &list);
|
2026-02-20 08:59:18 +00:00
|
|
|
if (error != SUCCESS) {
|
|
|
|
|
qWarning() << "list_accounts: wallet FFI error" << error;
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
QJsonArray result;
|
|
|
|
|
for (uintptr_t i = 0; i < list.count; ++i) {
|
|
|
|
|
result.append(ffiAccountListEntryToJson(list.entries[i]));
|
2026-02-18 22:10:38 +01:00
|
|
|
}
|
2026-02-20 08:59:18 +00:00
|
|
|
wallet_ffi_free_account_list(&list);
|
|
|
|
|
return result;
|
2026-02-04 15:51:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// === Account Queries ===
|
|
|
|
|
|
2026-02-20 08:59:18 +00:00
|
|
|
QString LogosExecutionZoneWalletModule::get_balance(const QString& account_id_hex, const bool is_public) {
|
2026-02-18 21:23:16 +01:00
|
|
|
FfiBytes32 id{};
|
|
|
|
|
if (!hexToBytes32(account_id_hex, &id)) {
|
2026-02-20 08:59:18 +00:00
|
|
|
qWarning() << "get_balance: invalid account_id_hex";
|
|
|
|
|
return {};
|
2026-02-04 15:51:02 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-18 21:23:16 +01:00
|
|
|
uint8_t balance[16] = {0};
|
|
|
|
|
const WalletFfiError error = wallet_ffi_get_balance(walletHandle, &id, is_public, &balance);
|
2026-02-20 08:59:18 +00:00
|
|
|
if (error != SUCCESS) {
|
|
|
|
|
qWarning() << "get_balance: wallet FFI error" << error;
|
|
|
|
|
return {};
|
2026-02-18 21:23:16 +01:00
|
|
|
}
|
2026-02-20 08:59:18 +00:00
|
|
|
return bytesToHex(balance, 16);
|
2026-02-04 15:51:02 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-20 08:59:18 +00:00
|
|
|
QString LogosExecutionZoneWalletModule::get_account_public(const QString& account_id_hex) {
|
2026-02-18 21:23:16 +01:00
|
|
|
FfiBytes32 id{};
|
|
|
|
|
if (!hexToBytes32(account_id_hex, &id)) {
|
2026-02-20 08:59:18 +00:00
|
|
|
qWarning() << "get_account_public: invalid account_id_hex";
|
|
|
|
|
return {};
|
2026-02-18 21:23:16 +01:00
|
|
|
}
|
2026-02-18 22:10:38 +01:00
|
|
|
FfiAccount account{};
|
|
|
|
|
const WalletFfiError error = wallet_ffi_get_account_public(walletHandle, &id, &account);
|
2026-02-20 08:59:18 +00:00
|
|
|
if (error != SUCCESS) {
|
|
|
|
|
qWarning() << "get_account_public: wallet FFI error" << error;
|
|
|
|
|
return {};
|
2026-02-18 22:10:38 +01:00
|
|
|
}
|
2026-02-20 08:59:18 +00:00
|
|
|
QString result = ffiAccountToJson(account);
|
|
|
|
|
wallet_ffi_free_account_data(&account);
|
|
|
|
|
return result;
|
2026-02-04 15:51:02 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-20 08:59:18 +00:00
|
|
|
QString LogosExecutionZoneWalletModule::get_account_private(const QString& account_id_hex) {
|
2026-02-18 21:23:16 +01:00
|
|
|
FfiBytes32 id{};
|
|
|
|
|
if (!hexToBytes32(account_id_hex, &id)) {
|
2026-02-20 08:59:18 +00:00
|
|
|
qWarning() << "get_account_private: invalid account_id_hex";
|
|
|
|
|
return {};
|
2026-02-18 21:23:16 +01:00
|
|
|
}
|
2026-02-18 22:10:38 +01:00
|
|
|
FfiAccount account{};
|
|
|
|
|
const WalletFfiError error = wallet_ffi_get_account_private(walletHandle, &id, &account);
|
2026-02-20 08:59:18 +00:00
|
|
|
if (error != SUCCESS) {
|
|
|
|
|
qWarning() << "get_account_private: wallet FFI error" << error;
|
|
|
|
|
return {};
|
2026-02-18 22:10:38 +01:00
|
|
|
}
|
2026-02-20 08:59:18 +00:00
|
|
|
QString result = ffiAccountToJson(account);
|
|
|
|
|
wallet_ffi_free_account_data(&account);
|
|
|
|
|
return result;
|
2026-02-18 09:20:58 +00:00
|
|
|
}
|
|
|
|
|
|
2026-02-20 08:59:18 +00:00
|
|
|
QString LogosExecutionZoneWalletModule::get_public_account_key(const QString& account_id_hex) {
|
2026-02-18 21:23:16 +01:00
|
|
|
FfiBytes32 id{};
|
|
|
|
|
if (!hexToBytes32(account_id_hex, &id)) {
|
2026-02-20 08:59:18 +00:00
|
|
|
qWarning() << "get_public_account_key: invalid account_id_hex";
|
|
|
|
|
return {};
|
2026-02-18 21:23:16 +01:00
|
|
|
}
|
|
|
|
|
FfiPublicAccountKey key{};
|
|
|
|
|
const WalletFfiError error = wallet_ffi_get_public_account_key(walletHandle, &id, &key);
|
2026-02-20 08:59:18 +00:00
|
|
|
if (error != SUCCESS) {
|
|
|
|
|
qWarning() << "get_public_account_key: wallet FFI error" << error;
|
|
|
|
|
return {};
|
2026-02-18 21:23:16 +01:00
|
|
|
}
|
2026-02-20 08:59:18 +00:00
|
|
|
return bytes32ToHex(key.public_key);
|
2026-02-04 15:51:02 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-20 08:59:18 +00:00
|
|
|
QString LogosExecutionZoneWalletModule::get_private_account_keys(const QString& account_id_hex) {
|
2026-02-18 21:23:16 +01:00
|
|
|
FfiBytes32 id{};
|
|
|
|
|
if (!hexToBytes32(account_id_hex, &id)) {
|
2026-02-20 08:59:18 +00:00
|
|
|
qWarning() << "get_private_account_keys: invalid account_id_hex";
|
|
|
|
|
return {};
|
2026-02-18 21:23:16 +01:00
|
|
|
}
|
2026-02-18 22:10:38 +01:00
|
|
|
FfiPrivateAccountKeys keys{};
|
|
|
|
|
const WalletFfiError error = wallet_ffi_get_private_account_keys(walletHandle, &id, &keys);
|
2026-02-20 08:59:18 +00:00
|
|
|
if (error != SUCCESS) {
|
|
|
|
|
qWarning() << "get_private_account_keys: wallet FFI error" << error;
|
|
|
|
|
return {};
|
2026-02-18 22:10:38 +01:00
|
|
|
}
|
2026-02-20 08:59:18 +00:00
|
|
|
QString result = ffiPrivateAccountKeysToJson(keys);
|
|
|
|
|
wallet_ffi_free_private_account_keys(&keys);
|
|
|
|
|
return result;
|
2026-02-04 15:51:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// === Account Encoding ===
|
|
|
|
|
|
2026-02-18 21:23:16 +01:00
|
|
|
QString LogosExecutionZoneWalletModule::account_id_to_base58(const QString& account_id_hex) {
|
|
|
|
|
FfiBytes32 id{};
|
|
|
|
|
if (!hexToBytes32(account_id_hex, &id)) {
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char* str = wallet_ffi_account_id_to_base58(&id);
|
2026-02-04 15:51:02 +01:00
|
|
|
if (!str) {
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString result = QString::fromUtf8(str);
|
|
|
|
|
wallet_ffi_free_string(str);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-20 08:59:18 +00:00
|
|
|
QString LogosExecutionZoneWalletModule::account_id_from_base58(const QString& base58_str) {
|
2026-02-18 21:23:16 +01:00
|
|
|
FfiBytes32 id{};
|
2026-02-04 16:18:56 +01:00
|
|
|
const QByteArray utf8 = base58_str.toUtf8();
|
2026-02-18 21:23:16 +01:00
|
|
|
const WalletFfiError error = wallet_ffi_account_id_from_base58(utf8.constData(), &id);
|
2026-02-20 08:59:18 +00:00
|
|
|
if (error != SUCCESS) {
|
|
|
|
|
qWarning() << "account_id_from_base58: wallet FFI error" << error;
|
|
|
|
|
return {};
|
2026-02-18 21:23:16 +01:00
|
|
|
}
|
2026-02-20 08:59:18 +00:00
|
|
|
return bytes32ToHex(id);
|
2026-02-04 15:51:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// === Blockchain Synchronisation ===
|
|
|
|
|
|
2026-02-18 11:21:00 +00:00
|
|
|
WalletFfiError LogosExecutionZoneWalletModule::sync_to_block(const uint64_t block_id) {
|
|
|
|
|
return wallet_ffi_sync_to_block(walletHandle, block_id);
|
2026-02-04 15:51:02 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-20 08:59:18 +00:00
|
|
|
uint64_t LogosExecutionZoneWalletModule::get_last_synced_block() {
|
|
|
|
|
uint64_t block_id = 0;
|
|
|
|
|
const WalletFfiError error = wallet_ffi_get_last_synced_block(walletHandle, &block_id);
|
|
|
|
|
if (error != SUCCESS) {
|
|
|
|
|
qWarning() << "get_last_synced_block: wallet FFI error" << error;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
return block_id;
|
2026-02-04 15:51:02 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-20 08:59:18 +00:00
|
|
|
uint64_t LogosExecutionZoneWalletModule::get_current_block_height() {
|
|
|
|
|
uint64_t block_height = 0;
|
|
|
|
|
const WalletFfiError error = wallet_ffi_get_current_block_height(walletHandle, &block_height);
|
|
|
|
|
if (error != SUCCESS) {
|
|
|
|
|
qWarning() << "get_current_block_height: wallet FFI error" << error;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
return block_height;
|
2026-02-04 15:51:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// === Operations ===
|
|
|
|
|
|
2026-02-20 08:59:18 +00:00
|
|
|
QString LogosExecutionZoneWalletModule::transfer_public(
|
2026-02-18 21:23:16 +01:00
|
|
|
const QString& from_hex,
|
|
|
|
|
const QString& to_hex,
|
2026-02-20 08:59:18 +00:00
|
|
|
const QString& amount_le16_hex
|
2026-02-04 15:51:02 +01:00
|
|
|
) {
|
2026-02-18 21:23:16 +01:00
|
|
|
FfiBytes32 fromId{}, toId{};
|
|
|
|
|
if (!hexToBytes32(from_hex, &fromId) || !hexToBytes32(to_hex, &toId)) {
|
2026-02-20 08:59:18 +00:00
|
|
|
qWarning() << "transfer_public: invalid account id hex";
|
|
|
|
|
return {};
|
2026-02-18 21:23:16 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-18 22:10:38 +01:00
|
|
|
uint8_t amount[16];
|
|
|
|
|
if (!hexToU128(amount_le16_hex, &amount)) {
|
2026-02-18 21:23:16 +01:00
|
|
|
qWarning() << "transfer_public: amount_le16_hex must be 32 hex characters (16 bytes)";
|
2026-02-20 08:59:18 +00:00
|
|
|
return {};
|
2026-02-04 15:51:02 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-18 22:10:38 +01:00
|
|
|
FfiTransferResult result{};
|
|
|
|
|
const WalletFfiError error = wallet_ffi_transfer_public(walletHandle, &fromId, &toId, &amount, &result);
|
2026-02-20 08:59:18 +00:00
|
|
|
if (error != SUCCESS) {
|
|
|
|
|
qWarning() << "transfer_public: wallet FFI error" << error;
|
|
|
|
|
return {};
|
2026-02-18 22:10:38 +01:00
|
|
|
}
|
2026-02-20 08:59:18 +00:00
|
|
|
QString resultJson = ffiTransferResultToJson(result);
|
|
|
|
|
wallet_ffi_free_transfer_result(&result);
|
|
|
|
|
return resultJson;
|
2026-02-04 15:51:02 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-20 08:59:18 +00:00
|
|
|
QString LogosExecutionZoneWalletModule::transfer_shielded(
|
2026-02-18 21:23:16 +01:00
|
|
|
const QString& from_hex,
|
2026-02-18 22:10:38 +01:00
|
|
|
const QString& to_keys_json,
|
2026-02-20 08:59:18 +00:00
|
|
|
const QString& amount_le16_hex
|
2026-02-18 09:20:58 +00:00
|
|
|
) {
|
2026-02-18 21:23:16 +01:00
|
|
|
FfiBytes32 fromId{};
|
|
|
|
|
if (!hexToBytes32(from_hex, &fromId)) {
|
2026-02-20 08:59:18 +00:00
|
|
|
qWarning() << "transfer_shielded: invalid from account id hex";
|
|
|
|
|
return {};
|
2026-02-18 21:23:16 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-18 22:10:38 +01:00
|
|
|
FfiPrivateAccountKeys toKeys{};
|
|
|
|
|
if (!jsonToFfiPrivateAccountKeys(to_keys_json, &toKeys)) {
|
2026-02-20 08:59:18 +00:00
|
|
|
qWarning() << "transfer_shielded: failed to parse to_keys_json";
|
|
|
|
|
return {};
|
2026-02-18 09:20:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t amount[16];
|
2026-02-18 22:10:38 +01:00
|
|
|
if (!hexToU128(amount_le16_hex, &amount)) {
|
|
|
|
|
qWarning() << "transfer_shielded: amount_le16_hex must be 32 hex characters (16 bytes)";
|
|
|
|
|
free(const_cast<uint8_t*>(toKeys.viewing_public_key));
|
2026-02-20 08:59:18 +00:00
|
|
|
return {};
|
2026-02-18 22:10:38 +01:00
|
|
|
}
|
2026-02-18 09:20:58 +00:00
|
|
|
|
2026-02-18 22:10:38 +01:00
|
|
|
FfiTransferResult result{};
|
|
|
|
|
const WalletFfiError error = wallet_ffi_transfer_shielded(walletHandle, &fromId, &toKeys, &amount, &result);
|
|
|
|
|
free(const_cast<uint8_t*>(toKeys.viewing_public_key));
|
2026-02-20 08:59:18 +00:00
|
|
|
if (error != SUCCESS) {
|
|
|
|
|
qWarning() << "transfer_shielded: wallet FFI error" << error;
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
QString resultJson = ffiTransferResultToJson(result);
|
|
|
|
|
wallet_ffi_free_transfer_result(&result);
|
|
|
|
|
return resultJson;
|
2026-02-18 09:20:58 +00:00
|
|
|
}
|
|
|
|
|
|
2026-02-20 08:59:18 +00:00
|
|
|
QString LogosExecutionZoneWalletModule::transfer_deshielded(
|
2026-02-18 21:23:16 +01:00
|
|
|
const QString& from_hex,
|
|
|
|
|
const QString& to_hex,
|
2026-02-20 08:59:18 +00:00
|
|
|
const QString& amount_le16_hex
|
2026-02-18 09:20:58 +00:00
|
|
|
) {
|
2026-02-18 21:23:16 +01:00
|
|
|
FfiBytes32 fromId{}, toId{};
|
|
|
|
|
if (!hexToBytes32(from_hex, &fromId) || !hexToBytes32(to_hex, &toId)) {
|
2026-02-20 08:59:18 +00:00
|
|
|
qWarning() << "transfer_deshielded: invalid account id hex";
|
|
|
|
|
return {};
|
2026-02-18 21:23:16 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-18 22:10:38 +01:00
|
|
|
uint8_t amount[16];
|
|
|
|
|
if (!hexToU128(amount_le16_hex, &amount)) {
|
2026-02-18 21:23:16 +01:00
|
|
|
qWarning() << "transfer_deshielded: amount_le16_hex must be 32 hex characters (16 bytes)";
|
2026-02-20 08:59:18 +00:00
|
|
|
return {};
|
2026-02-18 09:20:58 +00:00
|
|
|
}
|
|
|
|
|
|
2026-02-18 22:10:38 +01:00
|
|
|
FfiTransferResult result{};
|
|
|
|
|
const WalletFfiError error = wallet_ffi_transfer_deshielded(walletHandle, &fromId, &toId, &amount, &result);
|
2026-02-20 08:59:18 +00:00
|
|
|
if (error != SUCCESS) {
|
|
|
|
|
qWarning() << "transfer_deshielded: wallet FFI error" << error;
|
|
|
|
|
return {};
|
2026-02-18 22:10:38 +01:00
|
|
|
}
|
2026-02-20 08:59:18 +00:00
|
|
|
QString resultJson = ffiTransferResultToJson(result);
|
|
|
|
|
wallet_ffi_free_transfer_result(&result);
|
|
|
|
|
return resultJson;
|
2026-02-18 09:20:58 +00:00
|
|
|
}
|
|
|
|
|
|
2026-02-20 08:59:18 +00:00
|
|
|
QString LogosExecutionZoneWalletModule::transfer_private(
|
2026-02-18 21:23:16 +01:00
|
|
|
const QString& from_hex,
|
2026-02-18 22:10:38 +01:00
|
|
|
const QString& to_keys_json,
|
2026-02-20 08:59:18 +00:00
|
|
|
const QString& amount_le16_hex
|
2026-02-18 09:20:58 +00:00
|
|
|
) {
|
2026-02-18 21:23:16 +01:00
|
|
|
FfiBytes32 fromId{};
|
|
|
|
|
if (!hexToBytes32(from_hex, &fromId)) {
|
2026-02-20 08:59:18 +00:00
|
|
|
qWarning() << "transfer_private: invalid from account id hex";
|
|
|
|
|
return {};
|
2026-02-18 21:23:16 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-18 22:10:38 +01:00
|
|
|
FfiPrivateAccountKeys toKeys{};
|
|
|
|
|
if (!jsonToFfiPrivateAccountKeys(to_keys_json, &toKeys)) {
|
2026-02-20 08:59:18 +00:00
|
|
|
qWarning() << "transfer_private: failed to parse to_keys_json";
|
|
|
|
|
return {};
|
2026-02-18 09:20:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t amount[16];
|
2026-02-18 22:10:38 +01:00
|
|
|
if (!hexToU128(amount_le16_hex, &amount)) {
|
|
|
|
|
qWarning() << "transfer_private: amount_le16_hex must be 32 hex characters (16 bytes)";
|
|
|
|
|
free(const_cast<uint8_t*>(toKeys.viewing_public_key));
|
2026-02-20 08:59:18 +00:00
|
|
|
return {};
|
2026-02-18 22:10:38 +01:00
|
|
|
}
|
2026-02-18 09:20:58 +00:00
|
|
|
|
2026-02-18 22:10:38 +01:00
|
|
|
FfiTransferResult result{};
|
|
|
|
|
const WalletFfiError error = wallet_ffi_transfer_private(walletHandle, &fromId, &toKeys, &amount, &result);
|
|
|
|
|
free(const_cast<uint8_t*>(toKeys.viewing_public_key));
|
2026-02-20 08:59:18 +00:00
|
|
|
if (error != SUCCESS) {
|
|
|
|
|
qWarning() << "transfer_private: wallet FFI error" << error;
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
QString resultJson = ffiTransferResultToJson(result);
|
|
|
|
|
wallet_ffi_free_transfer_result(&result);
|
|
|
|
|
return resultJson;
|
2026-02-18 09:20:58 +00:00
|
|
|
}
|
|
|
|
|
|
2026-02-20 08:59:18 +00:00
|
|
|
QString LogosExecutionZoneWalletModule::transfer_shielded_owned(
|
2026-02-18 21:23:16 +01:00
|
|
|
const QString& from_hex,
|
|
|
|
|
const QString& to_hex,
|
2026-02-20 08:59:18 +00:00
|
|
|
const QString& amount_le16_hex
|
2026-02-18 09:20:58 +00:00
|
|
|
) {
|
2026-02-18 21:23:16 +01:00
|
|
|
FfiBytes32 fromId{}, toId{};
|
|
|
|
|
if (!hexToBytes32(from_hex, &fromId) || !hexToBytes32(to_hex, &toId)) {
|
2026-02-20 08:59:18 +00:00
|
|
|
qWarning() << "transfer_shielded_owned: invalid account id hex";
|
|
|
|
|
return {};
|
2026-02-18 21:23:16 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-18 22:10:38 +01:00
|
|
|
uint8_t amount[16];
|
|
|
|
|
if (!hexToU128(amount_le16_hex, &amount)) {
|
2026-02-18 21:23:16 +01:00
|
|
|
qWarning() << "transfer_shielded_owned: amount_le16_hex must be 32 hex characters (16 bytes)";
|
2026-02-20 08:59:18 +00:00
|
|
|
return {};
|
2026-02-18 09:20:58 +00:00
|
|
|
}
|
|
|
|
|
|
2026-02-18 22:10:38 +01:00
|
|
|
FfiTransferResult result{};
|
|
|
|
|
const WalletFfiError error = wallet_ffi_transfer_shielded_owned(walletHandle, &fromId, &toId, &amount, &result);
|
2026-02-20 08:59:18 +00:00
|
|
|
if (error != SUCCESS) {
|
|
|
|
|
qWarning() << "transfer_shielded_owned: wallet FFI error" << error;
|
|
|
|
|
return {};
|
2026-02-18 22:10:38 +01:00
|
|
|
}
|
2026-02-20 08:59:18 +00:00
|
|
|
QString resultJson = ffiTransferResultToJson(result);
|
|
|
|
|
wallet_ffi_free_transfer_result(&result);
|
|
|
|
|
return resultJson;
|
2026-02-18 09:20:58 +00:00
|
|
|
}
|
|
|
|
|
|
2026-02-20 08:59:18 +00:00
|
|
|
QString LogosExecutionZoneWalletModule::transfer_private_owned(
|
2026-02-18 21:23:16 +01:00
|
|
|
const QString& from_hex,
|
|
|
|
|
const QString& to_hex,
|
2026-02-20 08:59:18 +00:00
|
|
|
const QString& amount_le16_hex
|
2026-02-18 09:20:58 +00:00
|
|
|
) {
|
2026-02-18 21:23:16 +01:00
|
|
|
FfiBytes32 fromId{}, toId{};
|
|
|
|
|
if (!hexToBytes32(from_hex, &fromId) || !hexToBytes32(to_hex, &toId)) {
|
2026-02-20 08:59:18 +00:00
|
|
|
qWarning() << "transfer_private_owned: invalid account id hex";
|
|
|
|
|
return {};
|
2026-02-18 21:23:16 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-18 22:10:38 +01:00
|
|
|
uint8_t amount[16];
|
|
|
|
|
if (!hexToU128(amount_le16_hex, &amount)) {
|
2026-02-18 21:23:16 +01:00
|
|
|
qWarning() << "transfer_private_owned: amount_le16_hex must be 32 hex characters (16 bytes)";
|
2026-02-20 08:59:18 +00:00
|
|
|
return {};
|
2026-02-18 09:20:58 +00:00
|
|
|
}
|
|
|
|
|
|
2026-02-18 22:10:38 +01:00
|
|
|
FfiTransferResult result{};
|
|
|
|
|
const WalletFfiError error = wallet_ffi_transfer_private_owned(walletHandle, &fromId, &toId, &amount, &result);
|
2026-02-20 08:59:18 +00:00
|
|
|
if (error != SUCCESS) {
|
|
|
|
|
qWarning() << "transfer_private_owned: wallet FFI error" << error;
|
|
|
|
|
return {};
|
2026-02-18 22:10:38 +01:00
|
|
|
}
|
2026-02-20 08:59:18 +00:00
|
|
|
QString resultJson = ffiTransferResultToJson(result);
|
|
|
|
|
wallet_ffi_free_transfer_result(&result);
|
|
|
|
|
return resultJson;
|
2026-02-18 09:20:58 +00:00
|
|
|
}
|
|
|
|
|
|
2026-02-20 08:59:18 +00:00
|
|
|
QString LogosExecutionZoneWalletModule::register_public_account(const QString& account_id_hex) {
|
2026-02-18 21:23:16 +01:00
|
|
|
FfiBytes32 id{};
|
|
|
|
|
if (!hexToBytes32(account_id_hex, &id)) {
|
2026-02-20 08:59:18 +00:00
|
|
|
qWarning() << "register_public_account: invalid account_id_hex";
|
|
|
|
|
return {};
|
2026-02-18 21:23:16 +01:00
|
|
|
}
|
2026-02-18 22:10:38 +01:00
|
|
|
FfiTransferResult result{};
|
|
|
|
|
const WalletFfiError error = wallet_ffi_register_public_account(walletHandle, &id, &result);
|
2026-02-20 08:59:18 +00:00
|
|
|
if (error != SUCCESS) {
|
|
|
|
|
qWarning() << "register_public_account: wallet FFI error" << error;
|
|
|
|
|
return {};
|
2026-02-18 22:10:38 +01:00
|
|
|
}
|
2026-02-20 08:59:18 +00:00
|
|
|
QString resultJson = ffiTransferResultToJson(result);
|
|
|
|
|
wallet_ffi_free_transfer_result(&result);
|
|
|
|
|
return resultJson;
|
2026-02-04 15:51:02 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-20 08:59:18 +00:00
|
|
|
QString LogosExecutionZoneWalletModule::register_private_account(const QString& account_id_hex) {
|
2026-02-18 21:23:16 +01:00
|
|
|
FfiBytes32 id{};
|
|
|
|
|
if (!hexToBytes32(account_id_hex, &id)) {
|
2026-02-20 08:59:18 +00:00
|
|
|
qWarning() << "register_private_account: invalid account_id_hex";
|
|
|
|
|
return {};
|
2026-02-18 21:23:16 +01:00
|
|
|
}
|
2026-02-18 22:10:38 +01:00
|
|
|
FfiTransferResult result{};
|
|
|
|
|
const WalletFfiError error = wallet_ffi_register_private_account(walletHandle, &id, &result);
|
2026-02-20 08:59:18 +00:00
|
|
|
if (error != SUCCESS) {
|
|
|
|
|
qWarning() << "register_private_account: wallet FFI error" << error;
|
|
|
|
|
return {};
|
2026-02-18 22:10:38 +01:00
|
|
|
}
|
2026-02-20 08:59:18 +00:00
|
|
|
QString resultJson = ffiTransferResultToJson(result);
|
|
|
|
|
wallet_ffi_free_transfer_result(&result);
|
|
|
|
|
return resultJson;
|
2026-02-18 09:20:58 +00:00
|
|
|
}
|
|
|
|
|
|
2026-02-04 15:51:02 +01:00
|
|
|
// === Wallet Lifecycle ===
|
|
|
|
|
|
2026-02-18 11:21:00 +00:00
|
|
|
WalletFfiError LogosExecutionZoneWalletModule::create_new(
|
2026-02-04 15:51:02 +01:00
|
|
|
const QString& config_path,
|
|
|
|
|
const QString& storage_path,
|
|
|
|
|
const QString& password
|
|
|
|
|
) {
|
2026-02-18 11:21:00 +00:00
|
|
|
if (walletHandle) {
|
|
|
|
|
qWarning() << "create_new: wallet is already open";
|
|
|
|
|
return INTERNAL_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-04 16:18:56 +01:00
|
|
|
const QByteArray config_utf8 = config_path.toUtf8();
|
|
|
|
|
const QByteArray storage_utf8 = storage_path.toUtf8();
|
|
|
|
|
const QByteArray password_utf8 = password.toUtf8();
|
2026-02-04 15:51:02 +01:00
|
|
|
|
2026-02-18 11:21:00 +00:00
|
|
|
walletHandle = wallet_ffi_create_new(config_utf8.constData(), storage_utf8.constData(), password_utf8.constData());
|
|
|
|
|
if (!walletHandle) {
|
|
|
|
|
qWarning() << "create_new: wallet_ffi_create_new returned null";
|
|
|
|
|
return INTERNAL_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return SUCCESS;
|
2026-02-04 15:51:02 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-18 11:21:00 +00:00
|
|
|
WalletFfiError LogosExecutionZoneWalletModule::open(const QString& config_path, const QString& storage_path) {
|
|
|
|
|
if (walletHandle) {
|
|
|
|
|
qWarning() << "open: wallet is already open";
|
|
|
|
|
return INTERNAL_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-04 16:18:56 +01:00
|
|
|
const QByteArray config_utf8 = config_path.toUtf8();
|
|
|
|
|
const QByteArray storage_utf8 = storage_path.toUtf8();
|
2026-02-04 15:51:02 +01:00
|
|
|
|
2026-02-18 11:21:00 +00:00
|
|
|
walletHandle = wallet_ffi_open(config_utf8.constData(), storage_utf8.constData());
|
|
|
|
|
if (!walletHandle) {
|
|
|
|
|
qWarning() << "open: wallet_ffi_open returned null";
|
|
|
|
|
return INTERNAL_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return SUCCESS;
|
2026-02-04 15:51:02 +01:00
|
|
|
}
|
|
|
|
|
|
2026-02-18 11:21:00 +00:00
|
|
|
WalletFfiError LogosExecutionZoneWalletModule::save() {
|
|
|
|
|
return wallet_ffi_save(walletHandle);
|
2026-02-03 18:12:53 +01:00
|
|
|
}
|
2026-02-04 15:51:02 +01:00
|
|
|
|
|
|
|
|
// === Configuration ===
|
|
|
|
|
|
2026-02-18 11:21:00 +00:00
|
|
|
QString LogosExecutionZoneWalletModule::get_sequencer_addr() {
|
|
|
|
|
char* addr = wallet_ffi_get_sequencer_addr(walletHandle);
|
2026-02-04 15:51:02 +01:00
|
|
|
if (!addr) {
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString result = QString::fromUtf8(addr);
|
|
|
|
|
wallet_ffi_free_string(addr);
|
|
|
|
|
return result;
|
|
|
|
|
}
|