Payment button (widget) to allow users to pay for goods from tokens supported by Kyber, yet the merchants/ vendors can accept in whichever token they prefer.
The widget provides a friendly and convenient user interface for users to use ERC20 tokens to pay to an ETH address. Users can use different wallets of choice (for example, keystore, trezor, ledger, private key and metamask) to sign the transaction and make the payment, the widget will broadcast the transaction to the Ethereum network automatically and notify the app (vendors) about the transaction.
Above diagram shows how components like the widget, kyber network smart contract, vendor wallet, vendor servers interact to each other. Everything starts from end users.
As a vendor, what you care are:
- How can you use (setup, install) the widget to specify which token you will receive to which wallet, the payment amount,..etc. Read more
- How can you determine if a specific payment is pending, successful or failed (as the payment is failed to send, the payment is not sufficient or the payment is not in desired token). Read more
Vendor's backend server usually needs to monitor status of a payment in order to proceed to next step of an order (eg. sending confirmations to users, delivering the products, finalizing a deal...). With traditional payment gateways (Stripe, Paypal, Braintree...), vendors have to trust those gateways to send back the status, It is a bit different to KyberNetwork.
With KyberNetwork, the payment is made on blockchain thus vendors don't have to trust or rely on any data that comes from KyberNetwork but from the Ethereum network. However, vendors have to implement and run their own monitoring logic to get payment status from the Ethereum network or use and run the libs that KyberNetwork and the community have made.
Before broadcasting the transaction, the widget will send all infos about the payment (depends on how vendors configure via its params, read more) including transaction hash to the callback
(read more) passed to the widget. At this point, vendors' server is responsible to store that transaction hash and necessary payment infos (eg. user id, order id...) and use it to monitor that transaction's status on Ethereum network. Payment status will be determined based on that transaction status and the payment infos.
Basically, if the transaction is pending, payment is pending. If it is a reverted/failed tx, the payment is failed. If it is successful, the vendor needs to get the payment amount (in ETH or in ERC20 token) that the user sent and check if it is exactly the same to expected amount (price of a specific order).
There are 3 different transaction types the widget can broadcast.
- Pay in ETH, vendors get ETH. It is just a basic ETH transfer transaction.
- Pay in token A, vendors get token A. It is just a basic ERC20 transfer transaction.
- Pay in token A, vendors get token B (B != A). It is a
trade()
transaction
In order to get the amount of token the user has sent, the vendor needs to see if the transaction is which type. Each type will have different way to get token amount sent by the user as following:
- Transaction is the first type. Payment amount is transaction value.
- Transaction is the second type. Payment amount is logged in
event Transfer(address indexed _from, address indexed _to, uint _value)
, in_value
param. - Transaction is the third type. Payment amount is logged in
TradeExecute (index_topic_1 address origin, address src, uint256 srcAmount, address destToken, uint256 destAmount, address destAddress)
, indestAmount
param.
Pseudo code:
function paymentStatus(txhash, expectedPayment) -> return status {
startTime = time.Now()
txStatus = not_found
loop {
txStatus = getTxStatusFromEthereum(txhash) // possible returned value: not_found, pending, failed, success
switch txStatus {
case pending:
// wait more, do nothing
continue
case failed:
return "failed"
case success:
// TODO
case not_found:
if time.Now() - startTime > 15 mins {
// if the txhash is not found for more than 15 mins
return "failed"
}
}
}
}
First you should come to check KyberWidget Generator to generate all the params which is needed for your website. And then all you need to do is to copy & paste a snippet of code generated from there that renders a button with proper params linked to the widget.
Eg.
<a href="javascript:void(0);"
NAME="KyberPay - Powered by KyberNetwork" title="Pay by tokens"
onClick=window.open("https://developer.kyber.network/widget/payment?receiveAddr=0xFDF28Bf25779ED4cA74e958d54653260af604C20&receiveAmount=1.2&receiveToken=DAI&callback=https://yourwebsite.com/kybercallback&network=ropsten","Ratting","width=550,height=170,0,status=0");>Pay by tokens</a>
With that button, when a user click on it, a new window will pop up allowing him/her to do the payment. In this example, we passed several params to the widget via its url:
https://developer.kyber.network/widget/payment?receiveAddr=0xFDF28Bf25779ED4cA74e958d54653260af604C20&receiveAmount=1.2&receiveToken=DAI&callback=https://yourwebsite.com/kybercallback
that helps users to pay 1.2 DAI
equivalent amount of supported tokens (list of supported token will be given at the end of this Readme) to 0xFDF28Bf25779ED4cA74e958d54653260af604C20
(vendor's wallet), after the tx is broadcasted to the network, informations will be submitted to the callback url https://yourwebsite.com/kybercallback
.
In this version, we only support the widget via a new browser windows thus we can pass params via its url as url query params. The widget supports following params:
- receiveAddr (ethereum address with 0x prefix) - vendor's Ethereum wallet, user's payment will be sent there. Must double check this param very carefully. its value depend on field
type
- receiveToken (string) - token that you (vendor) want to receive, it can be one of supported tokens (such as ETH, DAI, KNC...). If you leave it blank or missing, the users can specify it in the widget interface
- receiveAmount (float) - the amount of
receiveToken
you (vendor) want your user to pay. If you leave it blank or missing, the users can specify it in the widget interface. It could be useful for undetermined payment or pay-as-you-go payment like a charity, ICO or anything else. - type (string) - default:
pay
. Possible value:pay, swap, buy
. Indicate which app will behave. If valuepay
, Interface will become payment,receiveAddr
is required. If valueswap
, Interface will be swap,receiveAddr
must be blank, other cases will cause error. If valuebuy
, Interface will be buy token,receiveAddr
must be blank,receiveToken
is required. - pinnedTokens (string) - default: "ETH_KNC_DAI". This param help to show priority tokens in list select token.
- defaultPair (string) - default: "ETH_KNC". This param only take effect when
type=swap
, it indicates default token pair will show in swap layout. - callback (string) - missing or blank value will prevent the widget to call the callback, the information will not be informed anywhere. Params submit to callback are in
urlencoded
format. - network (string) - default:
ropsten
, ethereum network that the widget will run. Possible value:test, ropsten, production, mainnet
.test
andropsten
will run on Ropsten network.production, mainnet
will run on Mainnet ethereum. Be carefull for this param! - paramForwarding (bool) - default:
true
, if it is true, all params that were passed to the widget will be submitted via thecallback
. It is useful that you can give your user a secret token (ideally one time token) to pass to the callback just so you know the callback is not coming from a malicious actor. - signer (string) - concatenation of a list of ethereum address by underscore
_
, eg. 0xFDF28Bf25779ED4cA74e958d54653260af604C20_0xFDF28Bf25779ED4cA74e958d54653260af604C20 - If you pass this param, the user will be forced to pay from one of those addresses. - commissionId - Ethereum address - your Ethereum wallet to get commission of the fees for the transaction.
- commissionFee - Commission Fee - Specified fee of the platform wallet. Value is set in Bps unit. If its not set, its consisdered as zero (default value).
- productName - String - You can push multiple
productName
to the URL in case there are more than 1 product - productQty - String - Just like
productName
, you can push multipleproductQty
to the URL *Note:productQty
always goes with aproductName
, it will be ignored if there is noproductName
- productImage - String - Just like
productName
, you can push multipleproductImage
to the URL *Note:productImage
always goes with aproductName
, it will be ignored if there is noproductName
- paymentData - String - A piece of additional information attached to the payment after broadcasted on the blockchain (*Note: This param only takes effect when type=pay)
- Pay mode with some product information:
https://widget.kyber.network/v0.6/?type=pay&mode=tab&receiveAddr=0x63B42a7662538A1dA732488c252433313396eade&receiveToken=KNC&callback=https%3A%2F%2Fkyberpay-sample.knstats.com%2Fcallback¶mForwarding=true&network=ropsten&receiveAmount=0.5&theme=theme-emerald&productName=A%20Cat%20Picture&productQty=7&productImage=https://images.unsplash.com/photo-1518791841217-8f162f1e1131&productName=Falling%20Autumn%20Leaves&productQty=24&productImage=https://images-wixmp-ed30a86b8c4ca887773594c2.wixmp.com/intermediary/f/fabdf38b-1811-49e8-8eeb-4c5632076c3e/dczgthc-076fcdf7-4932-4672-8d94-f3b6ed07d100.png&commissionId=0x90A21dbB74D7684B7AF747963D7ac7A8086b82B6
- Swap mode with Sunset theme
https://widget.kyber.network/v0.6/?type=swap&mode=tab&callback=https%3A%2F%2Fkyberpay-sample.knstats.com%2Fcallback¶mForwarding=true&network=ropsten&pinnedTokens=KNC_DAI&theme=theme-sunset
- Buy mode
https://widget.kyber.network/v0.6/?type=buy&mode=tab&receiveToken=ETH&receiveAmount=0.001&callback=https%3A%2F%2Fkyberpay-sample.knstats.com%2Fcallback¶mForwarding=true&network=ropsten&pinnedTokens=KNC_DAI&theme=theme-emerald
See all supported tokens here