I want to open up the conversation regarding account identifiers, principals, subaccounts and IcrcAccounts.
While talking with different ecosystem developers and users it’s appears that user controlled subaccounts are being integrated more and more by dapp developers, it seems that there is still a lot of confusion about the different “address types”.
Note that all data below is derived from the same data and all helper methods used can be found in different dfinity packages.
Principal
- widely adoptoped within the IC ecosystem
- does not support subaccounts
- recognizable format
import { Principal } from '@dfinity/principal';
const principal = Principal.fromText('wdarh-5xeo6-tbqmc-qmrpr-a5ppj-z4jbc-pprwe-gusfk-5qiaa-ocvfb-iqe');
// returns "wdarh-5xeo6-tbqmc-qmrpr-a5ppj-z4jbc-pprwe-gusfk-5qiaa-ocvfb-iqe"
const principalAsText = principal.toText();
//return "E477A6183050645F1075EF4E789089EF8D886A48AAEC10003855285102"
const principalAsHex = principal.toHex();
I don’t think
principal.toHex()
is actually used by developers but it does provide a valid hex that can be used which is the reason i added it.
AccountIdentifier
- added privacy
- not supported by most ICRC supported ledgers
- mainly used for transactions between the IC and centralized exchanges
- once converted it can’t be converted back (due to
SHA-224
)
import { AccountIdentifier, SubAccount } from '@dfinity/ledger-icp';
import { hexStringToUint8Array } from '@dfinity/utils';
const accountIdentifier = AccountIdentifier.fromPrincipal({ principal });
// returns "12f6a1ca374f7271ed2cc77470e5a7b1a649a21436367f898fbb639b3b12b21f"
const accountIdentifierhex = accountIdentifier.toHex()
const subaccount = SubAccount.fromBytes(hexStringToUint8Array(accountIdentifierhex)
// return "60cc844386ad1bd6747a7c024b63fbd610187e7d0910333863057410aefa4c1c"
const accountIdentifierWithSubaccount = AccountIdentifier.fromPrincipal({ principal, subaccount })
ICRCAccount
- widely supported within the ic ecosystem
- recognizable format
- supports subaccounts
- allows encoding and decoding to get the principal and subaccount
import { decodeIcrcAccount, encodeIcrcAccount } from '@dfinity/ledger-icrc';
import { principalToSubAccount } from '@dfinity/utils';
// returns "wdarh-5xeo6-tbqmc-qmrpr-a5ppj-z4jbc-pprwe-gusfk-5qiaa-ocvfb-iqe"
const icrcAccount = encodeIcrcAccount({ owner: principal });
// returns "wdarh-5xeo6-tbqmc-qmrpr-a5ppj-z4jbc-pprwe-gusfk-5qiaa-ocvfb-iqe-fwv5aqy.1de477a6183050645f1075ef4e789089ef8d886a48aaec100038552851020000"
const icrcAccountWithSubaccount = encodeIcrcAccount({ owner: principal, subaccount: principalToSubAccount(principal)})
If you want to read more about icrc account textual encode
I’m personally a fan of the ICRC account (textual encoding) because;
- a principal is also a valid ICRC account (without a subaccount)
- it allows the user to copy-paste the address only one time instead of a separate principal and subaccount
- it remain somewhat recognizable to the user.
For convenience we support the traditional and ICRC method on toolkit, but ideally we would remove manual in favor of ICRC accounts but this requires a widely adopted standard.
An other thing that i think could cause confusion and adoption problems is the naming, should it be called an icrcAccount
which the user need to learn do a deep dive into what it is?