2021-06-12 14:30:58 +03:00
|
|
|
// Copyright © 2021 Encrypted Ink. All rights reserved.
|
|
|
|
|
|
|
|
import Foundation
|
2021-06-12 18:22:02 +03:00
|
|
|
import Web3Swift
|
|
|
|
import CryptoSwift
|
2021-06-12 14:30:58 +03:00
|
|
|
|
|
|
|
struct Ethereum {
|
2021-06-13 13:42:42 +03:00
|
|
|
|
|
|
|
enum Errors: Error {
|
|
|
|
case invalidInputData
|
|
|
|
case failedToSendTransaction
|
|
|
|
}
|
2021-06-27 14:13:34 +03:00
|
|
|
|
|
|
|
private static let queue = DispatchQueue(label: "Ethereum", qos: .default)
|
2021-06-12 14:30:58 +03:00
|
|
|
|
2021-06-13 13:17:13 +03:00
|
|
|
private static let network: Network = AlchemyNetwork(
|
2021-06-12 18:22:02 +03:00
|
|
|
chain: "mainnet",
|
2021-06-21 23:56:35 +03:00
|
|
|
apiKey: Secrets.alchemy
|
2021-06-12 18:22:02 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
static func sign(message: String, account: Account) throws -> String {
|
|
|
|
let ethPrivateKey = EthPrivateKey(hex: account.privateKey)
|
2021-06-12 14:30:58 +03:00
|
|
|
|
2021-06-12 18:22:02 +03:00
|
|
|
let signature = SECP256k1Signature(
|
|
|
|
privateKey: ethPrivateKey,
|
|
|
|
message: UTF8StringBytes(string: message),
|
|
|
|
hashFunction: SHA3(variant: .keccak256).calculate
|
|
|
|
)
|
|
|
|
let data = try ConcatenatedBytes(
|
|
|
|
bytes: [
|
|
|
|
signature.r(),
|
|
|
|
signature.s(),
|
|
|
|
EthNumber(value: signature.recoverID().value() + 27)
|
|
|
|
]
|
|
|
|
).value()
|
|
|
|
return data.toPrefixedHexString()
|
2021-06-12 14:30:58 +03:00
|
|
|
}
|
|
|
|
|
2021-06-12 18:22:02 +03:00
|
|
|
static func signPersonal(message: String, account: Account) throws -> String {
|
|
|
|
let ethPrivateKey = EthPrivateKey(hex: account.privateKey)
|
|
|
|
let signed = SignedPersonalMessageBytes(message: message, signerKey: ethPrivateKey)
|
|
|
|
let data = try signed.value().toPrefixedHexString()
|
|
|
|
return data
|
2021-06-12 14:30:58 +03:00
|
|
|
}
|
|
|
|
|
2021-06-12 18:22:02 +03:00
|
|
|
static func sign(typedData: String, account: Account) throws -> String {
|
|
|
|
let data = try EIP712TypedData(jsonString: typedData)
|
|
|
|
let hash = EIP712Hash(domain: data.domain, typedData: data)
|
|
|
|
let privateKey = EthPrivateKey(hex: account.privateKey)
|
|
|
|
let signer = EIP712Signer(privateKey: privateKey)
|
|
|
|
return try signer.signatureData(hash: hash).toPrefixedHexString()
|
2021-06-12 14:30:58 +03:00
|
|
|
}
|
|
|
|
|
2021-06-13 13:17:13 +03:00
|
|
|
static func send(transaction: Transaction, account: Account) throws -> String {
|
2021-06-12 18:22:02 +03:00
|
|
|
let bytes = signedTransactionBytes(transaction: transaction, account: account)
|
2021-06-12 19:16:23 +03:00
|
|
|
let response = try SendRawTransactionProcedure(network: network, transactionBytes: bytes).call()
|
2021-06-13 13:17:13 +03:00
|
|
|
guard let hash = response["result"].string else {
|
2021-06-13 13:42:42 +03:00
|
|
|
throw Errors.failedToSendTransaction
|
2021-06-13 13:17:13 +03:00
|
|
|
}
|
|
|
|
return hash
|
2021-06-12 18:22:02 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
private static func signedTransactionBytes(transaction: Transaction, account: Account) -> EthContractCallBytes {
|
|
|
|
let senderKey = EthPrivateKey(hex: account.privateKey)
|
2021-06-20 16:17:11 +03:00
|
|
|
let contractAddress = EthAddress(hex: transaction.to)
|
|
|
|
let functionCall = BytesFromHexString(hex: transaction.data)
|
2021-06-13 13:17:13 +03:00
|
|
|
let bytes: EthContractCallBytes
|
|
|
|
if let gasPriceString = transaction.gasPrice {
|
|
|
|
let gasPrice = EthNumber(hex: gasPriceString)
|
2021-06-20 16:17:11 +03:00
|
|
|
if let gasEstimateString = transaction.gas,
|
|
|
|
let transctionCountString = transaction.nonce {
|
2021-06-13 13:17:13 +03:00
|
|
|
let gasEstimate = EthNumber(hex: gasEstimateString)
|
|
|
|
let transactionCount = EthNumber(hex: transctionCountString)
|
|
|
|
|
|
|
|
bytes = EthContractCallBytes(networkID: NetworkID(network: network),
|
|
|
|
transactionsCount: transactionCount,
|
|
|
|
gasPrice: gasPrice,
|
|
|
|
gasEstimate: gasEstimate,
|
|
|
|
senderKey: senderKey,
|
|
|
|
contractAddress: contractAddress,
|
2021-06-27 14:13:34 +03:00
|
|
|
weiAmount: transaction.weiAmount,
|
2021-06-13 13:17:13 +03:00
|
|
|
functionCall: functionCall)
|
|
|
|
} else {
|
|
|
|
bytes = EthContractCallBytes(network: network,
|
|
|
|
gasPrice: gasPrice,
|
|
|
|
senderKey: senderKey,
|
|
|
|
contractAddress: contractAddress,
|
2021-06-27 14:13:34 +03:00
|
|
|
weiAmount: transaction.weiAmount,
|
2021-06-13 13:17:13 +03:00
|
|
|
functionCall: functionCall)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
bytes = EthContractCallBytes(network: network,
|
|
|
|
senderKey: senderKey,
|
|
|
|
contractAddress: contractAddress,
|
2021-06-27 14:13:34 +03:00
|
|
|
weiAmount: transaction.weiAmount,
|
2021-06-13 13:17:13 +03:00
|
|
|
functionCall: functionCall)
|
|
|
|
}
|
2021-06-12 18:22:02 +03:00
|
|
|
return bytes
|
2021-06-12 14:30:58 +03:00
|
|
|
}
|
|
|
|
|
2021-06-27 14:13:34 +03:00
|
|
|
static func prepareTransaction(_ transaction: Transaction, completion: @escaping (Transaction) -> Void) {
|
|
|
|
var transaction = transaction
|
|
|
|
|
|
|
|
if transaction.nonce == nil {
|
|
|
|
getNonce(from: transaction.from) { nonce in
|
|
|
|
transaction.nonce = nonce
|
|
|
|
completion(transaction)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func getGasIfNeeded(gasPrice: String) {
|
|
|
|
guard transaction.gas == nil else { return }
|
|
|
|
getGas(from: transaction.from, to: transaction.to, gasPrice: gasPrice, weiAmount: transaction.weiAmount, data: transaction.data) { gas in
|
|
|
|
transaction.gas = gas
|
|
|
|
completion(transaction)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if let gasPrice = transaction.gasPrice {
|
|
|
|
getGasIfNeeded(gasPrice: gasPrice)
|
|
|
|
} else {
|
|
|
|
getGasPrice { gasPrice in
|
|
|
|
transaction.gasPrice = gasPrice
|
|
|
|
completion(transaction)
|
|
|
|
if let gasPrice = gasPrice {
|
|
|
|
getGasIfNeeded(gasPrice: gasPrice)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
private static func getGas(from: String, to: String, gasPrice: String, weiAmount: EthNumber, data: String, completion: @escaping (String?) -> Void) {
|
|
|
|
queue.async {
|
|
|
|
let gas = try? EthGasEstimate(
|
2021-06-26 01:07:15 +03:00
|
|
|
network: network,
|
|
|
|
senderAddress: EthAddress(hex: from),
|
|
|
|
recipientAddress: EthAddress(hex: to),
|
2021-06-27 14:13:34 +03:00
|
|
|
gasEstimate: EthGasEstimate(
|
|
|
|
network: network,
|
|
|
|
senderAddress: EthAddress(hex: from),
|
|
|
|
recipientAddress: EthAddress(hex: to),
|
|
|
|
gasPrice: EthNumber(hex: gasPrice),
|
|
|
|
weiAmount: weiAmount,
|
|
|
|
contractCall: BytesFromHexString(hex: data)
|
|
|
|
),
|
|
|
|
gasPrice: EthNumber(hex: gasPrice),
|
2021-06-26 01:07:15 +03:00
|
|
|
weiAmount: weiAmount,
|
|
|
|
contractCall: BytesFromHexString(hex: data)
|
2021-06-27 14:13:34 +03:00
|
|
|
).value().toHexString()
|
|
|
|
DispatchQueue.main.async {
|
|
|
|
completion(gas)
|
|
|
|
}
|
|
|
|
}
|
2021-06-26 01:07:15 +03:00
|
|
|
}
|
|
|
|
|
2021-06-27 14:13:34 +03:00
|
|
|
private static func getGasPrice(completion: @escaping (String?) -> Void) {
|
|
|
|
queue.async {
|
|
|
|
let gasPrice = try? EthGasPrice(network: network).value().toHexString()
|
|
|
|
DispatchQueue.main.async {
|
|
|
|
completion(gasPrice)
|
|
|
|
}
|
|
|
|
}
|
2021-06-26 01:07:15 +03:00
|
|
|
}
|
|
|
|
|
2021-06-27 14:13:34 +03:00
|
|
|
private static func getNonce(from: String, completion: @escaping (String?) -> Void) {
|
|
|
|
queue.async {
|
|
|
|
let nonce = try? EthTransactions(network: network, address: EthAddress(hex: from), blockChainState: PendingBlockChainState()).count().value().toHexString()
|
|
|
|
DispatchQueue.main.async {
|
|
|
|
completion(nonce)
|
|
|
|
}
|
|
|
|
}
|
2021-06-26 01:07:15 +03:00
|
|
|
}
|
|
|
|
|
2021-06-12 14:30:58 +03:00
|
|
|
}
|