Enable Canisters to Hold ICP

Tested it. There are still errors.

dfx canister call ledger_transfer canisterAddress '()'
The Replica returned an error: code 3, message: "Canister r7inp-6aaaa-aaaaa-aaabq-cai has no update method 'canisterAddress'"

It may be worth changing the dfx canister call ledger_transfer canisterAddress '()' command to dfx canister call ledger_transfer canisterAccount '()'. It seems to be working with her.
P.S. Sorry for my bad English :grinning:

2 Likes

This is the issue that Iā€™m addressing in the Invoice Canister payments flow initiative

2 Likes

Looking at the NNS ledger candid interface, it seems that there still isnā€™t a way to query a block if youā€™re using Candid/Motoko: Principal ryjl3-tyaaa-aaaaa-aaaba-cai | ic.rocks

Sorry if Iā€™m missing something, but it seems that I still need to trust the workaround canister (ockk2-xaaaa-aaaai-aaaua-cai) in order to validate transactions on chain?

1 Like

After doing some digging, the pattern Roman describes in his Community Conversation on canisters holding ICP provides what seems to be a good method for conducting payments onchain (my use case) without querying blocks.

3 Likes

I got an error. Anyon can help?

Many thanks,

Fwiw, I have pinged team to see if anyone can help with your error.

2 Likes

Many thanks, I used IC_VERSION=a7058d009494bea7e1d898a3dd7b525922979039

The error seems to indicate that your ledger.did file contains 404: Not Found, which is not valid Candid. I guess there was an error when you fetched it. What command did you use to fetch this file?

1 Like

Hello! Is there an instruction for setting up a canister that needs to be connected to ledger canister and manage icp on the main network?

1 Like

@bogwar Can the cycles minting canister give back the specific number of the cycles that it creates when topping up a canister? I know we can call for the exchange rate and make the calculation but it is possible that the exchange rate will change between the top-up call and the check-rate call, so it is better if the top-up call gives-back the number of the cycles that it mints.

Hi @levi,

Yes, I think we can return that info. Iā€™ll added to the things to change in the new for notifications.

1 Like

Hi @bogwar, Great thanks!

Reposting this from another thread for visibility: I need to be notified in my canister when someone pays ICP to it. Is some kind of subscription or event handling feature already implemented? I need this in a grant that just got approved. We were under the assumption that this was implemented already, but I canā€™t find anything about it.

Thereā€™s some work on that here by the open-chat team. Maybe this works for you?

1 Like

It is one of the biggest ā€œissuesā€ right now. You can do this by using transaction_notification method in the canister to which the payment that goes into the ledger is initially sent.
This would be a general workflow if going this route:

  1. User calls send_dfx(memo, amount, fee, from_subaccount, to, created_at_time) on Ledger to send ICP to our receiving canister (weā€™ll break down the parameters of that function call shortly.)
  2. User calls notify_dfx(block_height, max_fee, from_subaccount, to_canister, to_subaccount) .
  3. TxAuth receives the notification from Ledger that was created by our notify_dfx call. We validate the data we are receiving from the Ledger, and add it to our list of authenticated transactions.

There is an example of this in this repository GitHub - sagacards/legends-minter
However, example does not cover any checking of the actual ledger transaction in the canister inside of the transaction_notification method (which you must do to confirm if a payment was properly sent with correct amount to your canister).

Takeaways and issues by using this approach are:

  1. On the client side (javascript), user needs to send two transactions.
    First is to send_dfx method of the ledger canister to send ICP for whatever he wants to purchase
    Second to notify_dfx ledger canister method after his first transaction was a success (where he would get the actual block number as response that he must use in this call)

If a user navigates away from the client dapp before second transaction has been sent, you would never get transaction_notification called in your canister that got those ICPā€™s that the user sent in the first transaction, hence not actually being able to check it and send the user whatever he has purchased by his first transaction. This would lead to you having to issue refunds or similar to people that did not complete the full process and / or perhaps state that there are no refunds for the purchases made if users do not complete the complete ā€œcheckoutā€ processā€¦

You also need to have a bunch of bloated logic that saves what the user is trying to purchase BEFORE he does the first send_dfx transaction (ie: save his principal or similar and the ID of the item he is trying to purchase so that you can compare the item with that ID price vs how much was sent in that transaction later on in the notification_transaction method once you get a confirmation of a payment back from the ledger. You also need to take care of ā€œdouble spendā€ attack perhaps, where user might call notify_dfx with the same block number multiple times, which would make the transaction_notification trigger multiple times for the same transfer.

3 Likes

thatā€™s a cool project, but sadly, it seems it is no longer maintained and I also couldnā€™t find transaction verification logic in there.

(messed up the first post, didnā€™t link it to yours, so Iā€™m reposting).

Thanks for the quick reply. I had hoped there could be a simple function ā€œon_receiveā€ or something, that would be called on a canister (if the function exists), directly called by the asset holder (ICP or others), and would thus no longer need checking, as the caller would identify the asset type. This 2-step workflow and manual verification of transaction receipts is quite the bummer, to be honest.

I think I will build our canister so that it will require the asset holder canister to emit such an event to the receiver canister. Since the caller field in the notification call should be the asset holder, the source, token type, and integrity of the data is clearly verifiable by just checking the caller principal. Thus, we will not be supporting native ICP tokens, unless the ICP canister adds such notification functionality, or someone builds a wrapped ICP ledger that supports this kind of notification. I will post the exact specifications of the token standard Iā€™m proposing as a reply to this post once itā€™s done.

But the ledger canister DOES have that sort of notification mechanism already like I explained above, only, itā€™s a hassle to use and not without issues. You donā€™t need example of verification logic in that repository to use it, the verification logic can be done by you, depending on your needs.
Letā€™s say you are selling an NFT in your canister for 20 ICP. First, user would call some method in your canister called init_purchase or similar, where you would lock up that NFT so it cannot be purchased by anyone else in the meantime and also assign the callerā€™s principal (which is the buyer) to that NFT id somewhere in your smart contract (some kind of principal ā†’ nat pair, probably a hashmap). Then, user would (in the same flow on the front-end get a screen where he would be asked to pay 20 ICP (this is the first transaction I mentioned above). Once this is completed, you call one more tx (which he doesnā€™t need to know is happening as you can call it automatically i believe once the ledger tx is done for the sakes of better user experience) which would go to notify_dfx and you would send a block number that you get from the first successful transaction that you did when the user sent over 20 ICP to the ledger canister. Ledger canister would then automatically call your own canister method called transaction_notification with the details of the transaction from the ledger that corresponds to the block number that you sent to it in the notify_dfx call. Then, you can just compare what was the principal of the caller that made a tx to the ledger canister in that block, whether amount paid was actually 20 ICP (cost of that NFT), check if that principal is actually the same as the principal you stored in your custom init_purchase method and if it was, you could assign him that NFT, since he already paid for it.

A real hassle, but i think one of the ā€œeasierā€ ways of doing it

2 Likes

Thanks for the feedback, but Iā€™ve already made up my mind. The workflow you outlined only works for the ICP ledger and nothing else, I assume. Iā€™d rather have support for a generic token type and people would have to wrap their incompatible tokens than only support ICP in a non-portable way. Also, I donā€™t want to have to harden my canister against double spends, if I can avoid them by design. And I donā€™t think non-atomic transfers are acceptable, it will probably not fit in well with the client logic weā€™re using, and locking resources on the blockchain side is a huge error source. The code Iā€™m writing must be easily auditable and verifiable. Itā€™s not my fault the ICP ledger canister cannot be used conveniently in a black-box way.

2 Likes

You should participate in one of the Token standards workspaces and voice your concerns, better inform Dfinity about whats missing than adopt a community made standard that may be discontinued.

2 Likes