You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
375 lines
12 KiB
375 lines
12 KiB
//
|
|
// Created by artem on 24.01.18.
|
|
//
|
|
#include "etomiclib.h"
|
|
#include "etomiccurl.h"
|
|
#include <iostream>
|
|
#include <cpp-ethereum/libethcore/Common.h>
|
|
#include <cpp-ethereum/libethcore/CommonJS.h>
|
|
#include <cpp-ethereum/libethcore/TransactionBase.h>
|
|
#include <inttypes.h>
|
|
|
|
using namespace dev;
|
|
using namespace dev::eth;
|
|
|
|
char* stringStreamToChar(std::stringstream& ss)
|
|
{
|
|
const std::string tmp = ss.str();
|
|
auto result = (char*)malloc(strlen(tmp.c_str()) + 1);
|
|
strcpy(result, tmp.c_str());
|
|
return result;
|
|
}
|
|
|
|
TransactionSkeleton txDataToSkeleton(BasicTxData txData)
|
|
{
|
|
TransactionSkeleton tx;
|
|
tx.from = jsToAddress(txData.from);
|
|
tx.to = jsToAddress(txData.to);
|
|
tx.value = jsToU256(txData.amount);
|
|
tx.gas = 300000;
|
|
tx.gasPrice = 100 * exp10<9>();
|
|
tx.nonce = getNonce(txData.from);
|
|
return tx;
|
|
}
|
|
|
|
char* signTx(TransactionSkeleton& tx, char* secret)
|
|
{
|
|
Secret& secretKey = *(new Secret(secret));
|
|
auto baseTx = new TransactionBase(tx, secretKey);
|
|
RLPStream& rlpStream = *(new RLPStream());
|
|
baseTx->streamRLP(rlpStream);
|
|
std::stringstream& ss = *(new std::stringstream);
|
|
ss << rlpStream.out();
|
|
return stringStreamToChar(ss);
|
|
}
|
|
|
|
char* approveErc20(char* amount, char* from, char* secret)
|
|
{
|
|
TransactionSkeleton tx;
|
|
tx.from = jsToAddress(from);
|
|
tx.to = jsToAddress("0xc0eb7AeD740E1796992A08962c15661bDEB58003");
|
|
tx.value = 0; // exp10<18>();
|
|
tx.gas = 300000;
|
|
tx.gasPrice = 100 * exp10<9>();
|
|
tx.nonce = getNonce(from);
|
|
std::stringstream ss;
|
|
ss << "0x095ea7b3"
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress("0xe1D4236C5774D35Dc47dcc2E5E0CcFc463A3289c"))
|
|
<< toHex(toBigEndian(jsToU256(amount)));
|
|
tx.data = jsToBytes(ss.str());
|
|
char* rawTx = signTx(tx, secret);
|
|
char* result = sendRawTx(rawTx);
|
|
free(rawTx);
|
|
return result;
|
|
}
|
|
|
|
char* aliceSendsEthPayment(AliceSendsEthPaymentInput input, BasicTxData txData)
|
|
{
|
|
TransactionSkeleton tx = txDataToSkeleton(txData);
|
|
std::stringstream ss;
|
|
ss << "0x47c7b6e2"
|
|
<< toHex(jsToBytes(input.dealId))
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.bobAddress))
|
|
<< toHex(jsToBytes(input.aliceHash))
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToBytes(input.bobHash))
|
|
<< "000000000000000000000000";
|
|
tx.data = jsToBytes(ss.str());
|
|
char* rawTx = signTx(tx, txData.secretKey);
|
|
char* result = sendRawTx(rawTx);
|
|
free(rawTx);
|
|
return result;
|
|
}
|
|
|
|
char* aliceSendsErc20Payment(AliceSendsErc20PaymentInput input, BasicTxData txData)
|
|
{
|
|
TransactionSkeleton tx = txDataToSkeleton(txData);
|
|
std::stringstream ss;
|
|
ss << "0x184db3bf"
|
|
<< toHex(jsToBytes(input.dealId))
|
|
<< toHex(toBigEndian(jsToU256(input.amount)))
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.bobAddress))
|
|
<< toHex(jsToBytes(input.aliceHash))
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToBytes(input.bobHash))
|
|
<< "000000000000000000000000"
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.tokenAddress));
|
|
tx.data = jsToBytes(ss.str());
|
|
char* rawTx = signTx(tx, txData.secretKey);
|
|
char* result = sendRawTx(rawTx);
|
|
free(rawTx);
|
|
return result;
|
|
}
|
|
|
|
char* aliceReclaimsAlicePayment(AliceReclaimsAlicePaymentInput input, BasicTxData txData)
|
|
{
|
|
TransactionSkeleton tx = txDataToSkeleton(txData);
|
|
std::stringstream ss;
|
|
ss << "0x8b9a167a"
|
|
<< toHex(jsToBytes(input.dealId))
|
|
<< toHex(toBigEndian(jsToU256(input.amount)))
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.tokenAddress))
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.bobAddress))
|
|
<< toHex(jsToBytes(input.aliceHash))
|
|
<< "000000000000000000000000"
|
|
<< "00000000000000000000000000000000000000000000000000000000000000c0"
|
|
<< "0000000000000000000000000000000000000000000000000000000000000020"
|
|
<< toHex(jsToBytes(input.bobSecret));
|
|
tx.data = jsToBytes(ss.str());
|
|
char* rawTx = signTx(tx, txData.secretKey);
|
|
char* result = sendRawTx(rawTx);
|
|
free(rawTx);
|
|
return result;
|
|
}
|
|
|
|
char* bobSpendsAlicePayment(BobSpendsAlicePaymentInput input, BasicTxData txData)
|
|
{
|
|
TransactionSkeleton tx = txDataToSkeleton(txData);
|
|
std::stringstream ss;
|
|
ss << "0x392ec66b"
|
|
<< toHex(jsToBytes(input.dealId))
|
|
<< toHex(toBigEndian(jsToU256(input.amount)))
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.tokenAddress))
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.aliceAddress))
|
|
<< toHex(jsToBytes(input.bobHash))
|
|
<< "000000000000000000000000"
|
|
<< "00000000000000000000000000000000000000000000000000000000000000c0"
|
|
<< "0000000000000000000000000000000000000000000000000000000000000020"
|
|
<< toHex(jsToBytes(input.aliceSecret));
|
|
tx.data = jsToBytes(ss.str());
|
|
char* rawTx = signTx(tx, txData.secretKey);
|
|
char* result = sendRawTx(rawTx);
|
|
free(rawTx);
|
|
return result;
|
|
}
|
|
|
|
char* bobSendsEthDeposit(BobSendsEthDepositInput input, BasicTxData txData)
|
|
{
|
|
TransactionSkeleton tx = txDataToSkeleton(txData);
|
|
std::stringstream ss;
|
|
ss << "0xc2c5143f"
|
|
<< toHex(jsToBytes(input.depositId))
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.aliceAddress))
|
|
<< toHex(jsToBytes(input.bobHash))
|
|
<< "000000000000000000000000";
|
|
tx.data = jsToBytes(ss.str());
|
|
char* rawTx = signTx(tx, txData.secretKey);
|
|
char* result = sendRawTx(rawTx);
|
|
free(rawTx);
|
|
return result;
|
|
}
|
|
|
|
char* bobSendsErc20Deposit(BobSendsErc20DepositInput input, BasicTxData txData)
|
|
{
|
|
TransactionSkeleton tx = txDataToSkeleton(txData);
|
|
std::stringstream ss;
|
|
ss << "0xce8bbe4b"
|
|
<< toHex(jsToBytes(input.depositId))
|
|
<< toHex(toBigEndian(jsToU256(input.amount)))
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.aliceAddress))
|
|
<< toHex(jsToBytes(input.bobHash))
|
|
<< "000000000000000000000000"
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.tokenAddress));
|
|
tx.data = jsToBytes(ss.str());
|
|
char* rawTx = signTx(tx, txData.secretKey);
|
|
char* result = sendRawTx(rawTx);
|
|
free(rawTx);
|
|
return result;
|
|
}
|
|
|
|
char* bobRefundsDeposit(BobRefundsDepositInput input, BasicTxData txData)
|
|
{
|
|
TransactionSkeleton tx = txDataToSkeleton(txData);
|
|
std::stringstream ss;
|
|
ss << "0x1dbe6508"
|
|
<< toHex(jsToBytes(input.depositId))
|
|
<< toHex(toBigEndian(jsToU256(input.amount)))
|
|
<< toHex(toBigEndian(jsToU256(input.aliceCanClaimAfter)))
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.aliceAddress))
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.tokenAddress))
|
|
<< "00000000000000000000000000000000000000000000000000000000000000c0"
|
|
<< "0000000000000000000000000000000000000000000000000000000000000020"
|
|
<< toHex(jsToBytes(input.bobSecret));
|
|
tx.data = jsToBytes(ss.str());
|
|
char* rawTx = signTx(tx, txData.secretKey);
|
|
char* result = sendRawTx(rawTx);
|
|
free(rawTx);
|
|
return result;
|
|
}
|
|
|
|
char* aliceClaimsBobDeposit(AliceClaimsBobDepositInput input, BasicTxData txData)
|
|
{
|
|
TransactionSkeleton tx = txDataToSkeleton(txData);
|
|
std::stringstream ss;
|
|
ss << "0x960173b5"
|
|
<< toHex(jsToBytes(input.depositId))
|
|
<< toHex(toBigEndian(jsToU256(input.amount)))
|
|
<< toHex(toBigEndian(jsToU256(input.aliceCanClaimAfter)))
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.bobAddress))
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.tokenAddress))
|
|
<< toHex(jsToBytes(input.bobHash))
|
|
<< "000000000000000000000000";
|
|
tx.data = jsToBytes(ss.str());
|
|
char* rawTx = signTx(tx, txData.secretKey);
|
|
char* result = sendRawTx(rawTx);
|
|
free(rawTx);
|
|
return result;
|
|
}
|
|
|
|
char* bobSendsEthPayment(BobSendsEthPaymentInput input, BasicTxData txData)
|
|
{
|
|
TransactionSkeleton tx = txDataToSkeleton(txData);
|
|
std::stringstream ss;
|
|
ss << "0xcf36fe8e"
|
|
<< toHex(jsToBytes(input.paymentId))
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.aliceAddress))
|
|
<< toHex(jsToBytes(input.aliceHash))
|
|
<< "000000000000000000000000";
|
|
tx.data = jsToBytes(ss.str());
|
|
char* rawTx = signTx(tx, txData.secretKey);
|
|
char* result = sendRawTx(rawTx);
|
|
free(rawTx);
|
|
return result;
|
|
}
|
|
|
|
char* bobSendsErc20Payment(BobSendsErc20PaymentInput input, BasicTxData txData)
|
|
{
|
|
TransactionSkeleton tx = txDataToSkeleton(txData);
|
|
std::stringstream ss;
|
|
ss << "0x34f64dfd"
|
|
<< toHex(jsToBytes(input.paymentId))
|
|
<< toHex(toBigEndian(jsToU256(input.amount)))
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.aliceAddress))
|
|
<< toHex(jsToBytes(input.aliceHash))
|
|
<< "000000000000000000000000"
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.tokenAddress));
|
|
tx.data = jsToBytes(ss.str());
|
|
char* rawTx = signTx(tx, txData.secretKey);
|
|
char* result = sendRawTx(rawTx);
|
|
free(rawTx);
|
|
return result;
|
|
}
|
|
|
|
char* bobReclaimsBobPayment(BobReclaimsBobPaymentInput input, BasicTxData txData)
|
|
{
|
|
TransactionSkeleton tx = txDataToSkeleton(txData);
|
|
std::stringstream ss;
|
|
ss << "0xb7cc2312"
|
|
<< toHex(jsToBytes(input.paymentId))
|
|
<< toHex(toBigEndian(jsToU256(input.amount)))
|
|
<< toHex(toBigEndian(jsToU256(input.bobCanClaimAfter)))
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.aliceAddress))
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.tokenAddress))
|
|
<< toHex(jsToBytes(input.aliceHash))
|
|
<< "000000000000000000000000";
|
|
tx.data = jsToBytes(ss.str());
|
|
char* rawTx = signTx(tx, txData.secretKey);
|
|
char* result = sendRawTx(rawTx);
|
|
free(rawTx);
|
|
return result;
|
|
}
|
|
|
|
char* aliceSpendsBobPayment(AliceSpendsBobPaymentInput input, BasicTxData txData)
|
|
{
|
|
TransactionSkeleton tx = txDataToSkeleton(txData);
|
|
std::stringstream ss;
|
|
ss << "0x97004255"
|
|
<< toHex(jsToBytes(input.paymentId))
|
|
<< toHex(toBigEndian(jsToU256(input.amount)))
|
|
<< toHex(toBigEndian(jsToU256(input.bobCanClaimAfter)))
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.bobAddress))
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(input.tokenAddress))
|
|
<< "00000000000000000000000000000000000000000000000000000000000000c0"
|
|
<< "0000000000000000000000000000000000000000000000000000000000000020"
|
|
<< toHex(jsToBytes(input.aliceSecret));
|
|
tx.data = jsToBytes(ss.str());
|
|
char* rawTx = signTx(tx, txData.secretKey);
|
|
char* result = sendRawTx(rawTx);
|
|
free(rawTx);
|
|
return result;
|
|
}
|
|
|
|
char* privKey2Addr(char* privKey)
|
|
{
|
|
Secret& secretKey = *(new Secret(privKey));
|
|
std::stringstream& ss = *(new std::stringstream);
|
|
ss << "0x" << toAddress(secretKey);
|
|
return stringStreamToChar(ss);
|
|
};
|
|
|
|
char* pubKey2Addr(char* pubKey)
|
|
{
|
|
Public& publicKey = *(new Public(pubKey));
|
|
std::stringstream& ss = *(new std::stringstream);
|
|
ss << "0x" << toAddress(publicKey);
|
|
return stringStreamToChar(ss);
|
|
};
|
|
|
|
char* getPubKeyFromPriv(char* privKey)
|
|
{
|
|
Public publicKey = toPublic(*(new Secret(privKey)));
|
|
std::stringstream& ss = *(new std::stringstream);
|
|
ss << "0x" << publicKey;
|
|
return stringStreamToChar(ss);
|
|
}
|
|
|
|
uint64_t getEthBalance(char* address)
|
|
{
|
|
char* hexBalance = getEthBalanceRequest(address);
|
|
// convert wei to satoshi
|
|
u256 balance = jsToU256(hexBalance) / exp10<10>();
|
|
free(hexBalance);
|
|
return static_cast<uint64_t>(balance);
|
|
}
|
|
|
|
uint64_t getErc20Balance(char* address, char* tokenAddress)
|
|
{
|
|
std::stringstream ss;
|
|
ss << "0x70a08231"
|
|
<< "000000000000000000000000"
|
|
<< toHex(jsToAddress(address));
|
|
std::stringstream& resultStream = *(new std::stringstream);
|
|
char* hexBalance = ethCall(tokenAddress, ss.str().c_str());
|
|
// convert wei to satoshi
|
|
u256 balance = jsToU256(hexBalance) / exp10<10>();
|
|
free(hexBalance);
|
|
return static_cast<uint64_t>(balance);
|
|
}
|
|
|
|
void uint8arrayToHex(char *dest, uint8_t *input, int len)
|
|
{
|
|
strcpy(dest, "0x");
|
|
for (int i = 0; i < len; i++)
|
|
{
|
|
sprintf(dest + (i + 1) * 2, "%02x", input[i]);
|
|
}
|
|
dest[(len + 1) * 2] = '\0';
|
|
}
|
|
|
|
void satoshisToWei(char *dest, uint64_t input)
|
|
{
|
|
sprintf(dest, "%" PRIu64, input);
|
|
strcat(dest, "0000000000");
|
|
}
|
|
|