Skip to content

Commit

Permalink
payment: erc20next: add driver status events to debit notes & invoices (
Browse files Browse the repository at this point in the history
#2880)

* payment: add driver status events to payable debit notes & invoices
  * Fix details ser/de for events.
  * General refactor.
* erc20next: remove rpc.goerli.mudit.blog endpoint
* erc20next: update readme.md
* erc20next: add driver address to InsufficientGas and InsufficientToken statuses
* payment: fix driver status events for was_already_ok case
* payment: Add indexes on status columnt for pay_{invoice,debit_note}
  • Loading branch information
kamirr authored Nov 14, 2023
1 parent 5f969fa commit 74dfb7a
Show file tree
Hide file tree
Showing 18 changed files with 619 additions and 331 deletions.
8 changes: 4 additions & 4 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

6 changes: 3 additions & 3 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -224,7 +224,7 @@ members = [
# diesel 1.4.* supports up to 0.23.0, but sqlx 0.5.9 requires 0.22.0
# sqlx 0.5.10 need 0.23.2, so 0.5.9 is last version possible
libsqlite3-sys = { version = "0.26.0", features = ["bundled"] }
erc20_payment_lib = { git = "https://github.com/golemfactory/erc20_payment_lib", rev = "101e6bf331db3fd6b2b55813abc9ccccbc4960b7" }
erc20_payment_lib = { git = "https://github.com/golemfactory/erc20_payment_lib", rev = "f3559980c8998f2865e593be73766ad2abcc11ca" }
#erc20_payment_lib = { path = "../../payments/erc20_payment_lib/crates/erc20_payment_lib" }
#erc20_payment_lib = { version = "=0.3.1" }
rand = "0.8.5"
Expand Down Expand Up @@ -270,8 +270,8 @@ ya-sb-util = { git = "https://github.com/golemfactory/ya-service-bus.git", rev =
#ya-sb-util = { path = "../ya-service-bus/crates/util" }

## CLIENT
ya-client = { git = "https://github.com/golemfactory/ya-client.git", rev = "2bb679e3bb1d61eddd713d7f19ee127595f27162" }
ya-client-model = { git = "https://github.com/golemfactory/ya-client.git", rev = "2bb679e3bb1d61eddd713d7f19ee127595f27162" }
ya-client = { git = "https://github.com/golemfactory/ya-client.git", rev = "afdc1daf1f17a5128d911f8fff9d55c4f213e9af" }
ya-client-model = { git = "https://github.com/golemfactory/ya-client.git", rev = "afdc1daf1f17a5128d911f8fff9d55c4f213e9af" }

## RELAY and networking stack
ya-relay-stack = { git = "https://github.com/golemfactory/ya-relay.git", rev = "c92a75b0cf062fcc9dbb3ea2a034d913e5fad8e5" }
Expand Down
13 changes: 12 additions & 1 deletion core/model/src/payment.rs
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@ pub enum RpcMessageError {
}

pub mod local {
use super::*;
use super::{public::Ack, *};
use crate::driver::{AccountMode, GasDetails, PaymentConfirmation};
use bigdecimal::{BigDecimal, Zero};
use chrono::{DateTime, Utc};
Expand Down Expand Up @@ -424,6 +424,17 @@ pub mod local {
Internal(String),
}

#[derive(Clone, Debug, Serialize, Deserialize)]
pub struct PaymentDriverStatusChange {
pub properties: Vec<DriverStatusProperty>,
}

impl RpcMessage for PaymentDriverStatusChange {
const ID: &'static str = "PaymentDriverStatusChange";
type Item = Ack;
type Error = GenericError;
}

#[derive(Clone, Debug, Serialize, Deserialize)]
pub struct PaymentDriverStatus {
pub driver: Option<String>,
Expand Down
13 changes: 12 additions & 1 deletion core/payment-driver/base/src/bus.rs
Original file line number Diff line number Diff line change
Expand Up @@ -7,14 +7,15 @@
// External crates
use std::sync::Arc;

use ya_client_model::payment::DriverStatusProperty;
// Workspace uses
use ya_client_model::payment::driver_details::DriverDetails;
use ya_client_model::NodeId;
use ya_core_model::driver::{
driver_bus_id, AccountMode, GenericError, PaymentConfirmation, PaymentDetails,
};
use ya_core_model::identity;
use ya_core_model::payment::local as payment_srv;
use ya_core_model::payment::local::{self as payment_srv, PaymentDriverStatusChange};
use ya_service_bus::{
typed::{service, ServiceBinder},
RpcEndpoint,
Expand Down Expand Up @@ -191,3 +192,13 @@ pub async fn notify_payment(
.map_err(GenericError::new)?;
Ok(())
}

pub async fn status_changed(properties: Vec<DriverStatusProperty>) -> Result<(), GenericError> {
let msg = PaymentDriverStatusChange { properties };
service(payment_srv::BUS_ID)
.send(msg)
.await
.map_err(GenericError::new)?
.map_err(GenericError::new)?;
Ok(())
}
213 changes: 53 additions & 160 deletions core/payment-driver/erc20next/Readme.md
Original file line number Diff line number Diff line change
@@ -1,163 +1,56 @@
## Current ERC20 transactions flow
Date: 2021-10-22
# Erc20Next Payment driver
## Functionality
A payment driver is an abstraction over any operations relating to funds, which includes:
* Scheduling transfers to run at any point in the future.
* Verifying transfers done by other parties.
* Checking acount balance.
* Reporting status of scheduled transactions and the account.

Disclaimer: This is dev documentation not officially maintained, it is intended for internal development.

Testing transfers:
This command will send 0.0001 GLMs from internal wallet to address 0x89Ef977db64A2597bA57E3eb4b717D3bAAeBaeC3 (use your own address for testing)
Note that service have to be running otherwise you get no connection error.

```
yagna.exe payment transfer --amount 0.0001 --driver erc20next --network mumbai --to-address 0x89Ef977db64A2597bA57E3eb4b717D3bAAeBaeC3
```

You can specify extra options
* --gas-price (starting gas price in gwei)
* --max-gas-price (maximum allowed gas price in gwei)
* --gas-limit (limit of gas used in transaction). Better to leave default as it is not affecting cost of transaction. This is convenient for testing errors on blockchain.

```
yagna.exe payment transfer --amount 0.0001 --gas-price 1.1 --max-gas-price 60.4 --gas-limit 80000 --driver erc20next --network mumbai --to-address 0x89Ef977db64A2597bA57E3eb4b717D3bAAeBaeC3
```

Networks currently supported:
* mainnnet (ETH mainnet, do not use)
* rinkeby (ETH testnet, good support)
* goerli (ETH testnet)
* mumbai (Polygon testnet)
* polygon (Polygon mainnet)
The Erc20Next driver is such an abstraction built on top of the [ERC20 standard](https://ethereum.org/en/developers/docs/standards/tokens/erc-20/).

## Implementation

DB fields explained

```sql
tx_id TEXT NOT NULL PRIMARY KEY,
sender TEXT NOT NULL,
nonce INTEGER NOT NULL DEFAULT -1,
status INTEGER NOT NULL,
tx_type INTEGER NOT NULL,
tmp_onchain_txs TEXT NULL,
final_tx TEXT NULL,
starting_gas_price DOUBLE NULL,
current_gas_price DOUBLE NULL,
max_gas_price DOUBLE NULL,
final_gas_price DOUBLE NULL,
final_gas_used INTEGER NULL,
gas_limit INTEGER NULL,
time_created DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP,
time_last_action DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP,
time_sent DATETIME NULL,
time_confirmed DATETIME NULL,
network INTEGER NOT NULL DEFAULT 4,
last_error_msg TEXT NULL,
resent_times INT DEFAULT 0,
signature TEXT NULL,
encoded TEXT NOT NULL,
```

* tx_id - unique UUID4 generated for trasnsaction
* sender - internal yagna address used for sending transaction
* nonce - Ethereum nonce assigned to transaction
* status - status of the transaction:
* CREATED(1) - the transaction is submitted to db and wait to be sent
* SENT(2) - the transaction is successfully sent to the blockchain network
* PENDING(3) - the transaction is found on blockchain but waiting for execution
* CONFIRMED(4) - the transaction is confirmed and succeeded
* ERRORSENT(10) - transaction failed to be sent on-chain (not consuming nonce, has to be repeated)
* ERRORONCHAIN(11) - transaction is confirmed but failed on-chain (consuming nonce, cannot be repeated until new transaction is assigned to payment)
* tx_type (transfer or faucet transaction)
* tmp_onchain_txs - hashes of all transactions that are sent to the chain (important for checking transaction status when gas is increased)
* final_tx - onchain transaction hash only when transaction is CONFIRMED or ERRORONCHAIN. tmp_onchain_txs are removed to reduce clutter.
* starting_gas_price - gas in Gwei
* current_gas_price - starts with null then it has assigned higher level of gas until transaction is processed
* max_gas_price - limit for current_gas_price
* final_gas_price - assigned after transaction is CONFIRMED or ERRORONCHAIN.
* final_gas_used - assigned after transaction is CONFIRMED or ERRORONCHAIN. use final_gas_used * final_gas_price to have transaction cost in Gwei
* gas_limit - assigned max gas for transaction. If set to low ends with error during transaction sent or error on chain depending on set value.
* time_created - UTC time when entry is created.
* time_last_action - UTC time of last change of the entry.
* time_sent - UTC time of last succesfull sent.
* time_confirmed - UTC time of transaction confirmation (can be also when error on-chain)
* network - id of the Network (for example: 80001 is Mumbai, 137 is Polygon)
* last_error_msg - last error during sending or error onchain, Nulled when trasnaction is successfull
* resent_times - not used right now, intended to limit transaction retries
* signature - transaction signature
* encoded - YagnaRawTransaction encoded in json

## Assigning nonces

To process transaction in ethereum network you have to assign nonce which has to be strictly one greater than previous nonce.
This makes process of sending transaction tricky and complicated, because when you send two transactions with the same nonce one of them will fail. Other case is when one transaction is blocked or waiting others with higher nonce will get stuck too.
Currently for every transaction nonce is assigned and not changed until transaction will consume nonce on chain.

Huge issue: When transaction is malformed and it get stuck, resend will not help and all transactions are blocked.

With the implementation of the driver we are trying to resolve cases automatically but the process is complicated and not perfect right now.

Good news: Nonce prevents double spending so we don't have to worry about that (unless we change the nonce)

Note: Error on chain will consume nonce and gas and new nonce has to be assigned to proceed with transaction, which is currently not handled.

## Bumping gas prices

Problem on Ethereum network is as follows:
If you sent transaction you have only vague idea when transaction will be performed.
There is function that is estimating current gas price but it is far from perfect.
Also we want to pay for gas as little as possible.
For example on Polygon Network if you sent transaction for 30.01Gwei you can be pretty sure it get proceeded fairly smoothly
right now, but when the network is congested transaction can get stuck for indefinite amount of time.
When network usage is lower you can try to make transaction for lower gas fee (for example 20Gwei).
To resolve this issue we implemented automatic gas bumping. Every set amount of time gas is increased to increase probability of transaction getting processed.
That way we can save money on gas and also have bigger chance of transactions not getting stuck.

Currently we proposed following gas levels for polygon network:
```
10.011, //LOW PRIORITY
15.011, //LOW PRIORITY
20.011, //LOW PRIORITY
25.011, //LOW PRIORITY
30.011, //minimum suggested gas price - FAST PRIORITY
33.011, //not used
36.011, //not used
40.011, //not used
50.011, //not used
60.011, //EXPRESS PRIORITY
80.011,
100.011
```
Note that we add 0.011 to increase the chance of getting inline of someone setting whole number as gas price (which people tend to do). It costs almost nothing but significally increase your chance of transaction beeing processed.

Note that on test networks gas doesn't matter and transaction is processed instantly regardless of gas set. So to test this
feature you have to use Polygon network and pay some Matic for gas.

## VARIABLES:

POLYGON_PRIORITY:
possible values:
slow - normal, low priority, economic mode,
fast - fast transaction (for testing or normal mode)
express - express transaction (for testing)

ERC20_WAIT_FOR_PENDING_ON_NETWORK: (duration)
after that time transaction is resent with higher gas

## List of known errors:

Error when sending when gas-limit set too low
```
RPC error: Error { code: ServerError(-32000), message: "intrinsic gas too low", data: None }
```
```
RPC error: Error { code: ServerError(-32000), message: "already known", data: None }
```
```
RPC error: Error { code: ServerError(-32000), message: "nonce too low", data: None }
```







The core implementation is in [erc20_payment_lib](https://github.com/golemfactory/erc20_payment_lib), this crate only serves as an interface connecting
it to yagna.

## Configuration
### Via environment variables
#### Global settings
* `ERC20NEXT_SENDOUT_INTERVAL_SECS` -- The maximum interval at which transactions are batched and processed. A longer duration may conserve gas at the expense
of delivering payments at a later date.
#### Per-chain settings
In environment variables below, substitute `{CHAIN}` for the actual chain you wish to configure and `{GLM}` for the GLM symbol used on the chain.
To avoid confusion, `TGLM` is used on test chains that can mint GLM and `GLM` on non-test chains.
See `config-payments.toml` for the list of supported chains and token symbols.
* `{CHAIN}_GETH_ADDR` -- List of comma-separated RPC endpoints to be used.
* `{CHAIN}_PRIORITY_FEE` -- [priority fee](https://ethereum.org/nl/developers/docs/gas/#priority-fee).
* `{CHAIN}_MAX_FEE_PER_GAS` -- [max fee per gas](https://ethereum.org/nl/developers/docs/gas/#maxfee).
* `{CHAIN}_{SYMBOL}_CONTRACT_ADDRESS` -- Address of the GLM contract.
* `{CHAIN}_MULTI_PAYMENT_CONTRACT_ADDRESS` -- Address of a custom Golem contract allowing for executing multiple transfers at once.
* `ERC20NEXT_{CHAIN}_REQUIRED_CONFIRMATIONS` -- The number of confirmation blocks required to consider a transaction complete.

Be aware that options not prefixed with `ERC20NEXT` are also applicable to the old Erc20 driver.

### Via TOML file
* The default configuration can be seen in `config-payments.toml`.
* It can be overriden by placing a `config-payments.toml` file in yagna data directory. This is not recommended and is not guaranteed to work across versions.

## Statuses
The Erc20Next driver can report a selection of statuses which indicate possible issues.
* `InsufficientGas`:
* An account does not have sufficient gas to execute further transactions.
* Contains: `driver`, `network`, `address`, `neededGasEst`.
* `InsufficientToken`:
* An account does not have sufficient funds to execute further transactions.
* Contains: `driver`, `network`, `address`, `neededTokenEst`.
* `InvalidChainId`:
* A transaction has been scheduled on a chain that is not present in the configuration. This can only happen if `payments-config.toml` has been changed in an incorrect manner.
* Contains: `driver`, `chainId`.
* `CantSign`:
* The transaction cannot be signed. This means that yagna cannot access the identitiy used for this transfer, which can be caused by it being removed or locked.
* Contains: `driver`, `network`, `address`.
* `TxStuck`:
* A transaction cannot proceed despite being sent to the blockchain. The most likely reason is too low `max fee` setting.
* Contains: `driver`, `network`.
* `RpcError`:
* An RPC endpoint is unreliable. Consider using a better selection of endpoints.
* `driver`, `network`
12 changes: 7 additions & 5 deletions core/payment-driver/erc20next/config-payments.toml
Original file line number Diff line number Diff line change
Expand Up @@ -23,11 +23,12 @@ ignore-deadlines = false
[chain.goerli]
chain-name = "Goerli"
chain-id = 5
rpc-endpoints = ["https://ethereum-goerli-rpc.allthatnode.com",
"https://rpc.goerli.mudit.blog",
rpc-endpoints = [
"https://ethereum-goerli-rpc.allthatnode.com",
"https://rpc.slock.it/goerli",
"https://www.ethercluster.com/goerli",
"https://rpc.ankr.com/eth_goerli"]
"https://rpc.ankr.com/eth_goerli",
]
currency-symbol = "tETH"
priority-fee = 1.01
max-fee-per-gas = 300.0
Expand All @@ -41,7 +42,9 @@ block-explorer-url = "https://goerli.etherscan.io"
[chain.mumbai]
chain-name = "Mumbai testnet"
chain-id = 80001
rpc-endpoints = ["https://rpc-mumbai.maticvigil.com/v1/fd04db1066cae0f44d3461ae6d6a7cbbdd46e4a5"]
rpc-endpoints = [
"https://rpc-mumbai.maticvigil.com/v1/fd04db1066cae0f44d3461ae6d6a7cbbdd46e4a5",
]
# rpc-endpoints = ["http://127.0.0.1:8545"]
currency-symbol = "tMATIC"
priority-fee = 1.01
Expand All @@ -66,4 +69,3 @@ token = { address = "0x0B220b82F3eA3B7F6d9A1D8ab58930C064A2b5Bf", symbol = "GLM"
# multi-contract = { address = "0x50100d4faf5f3b09987dea36dc2eddd57a3e561b", max-at-once = 10 }
confirmation-blocks = 1
block-explorer-url = "https://polygonscan.com"

Loading

0 comments on commit 74dfb7a

Please sign in to comment.