Loyalty Tokens
Using the Sui Closed-Loop Token standard, you can create tokens that are valid only for a specific service, like an airline that wants to grant tokens to frequent flyers to purchase tickets or upgrades.
The following example demonstrates the creation of a loyalty token that bearers can use to make purchases in a digital gift shop.
Example
The Loyalty Token example illustrates a loyalty token that is created with the Closed Loop Token standard. If you were to implement this example, the Admin would send LOYALTY
tokens to the users of your service as a reward for their loyalty. The example creates a GiftShop
where holders can spend LOYALTY
tokens to buy Gift
s.
examples::loyalty
The loyalty.move source file contains the examples::loyalty
module code that creates the loyalty token. The module includes the one-time witness (OTW) that creates the coin (with the same name as the module, LOYALTY
), possesses only the drop
ability, and has no fields. These are the characteristics of a OTW, which ensures the LOYALTY
type has a single instance.
public struct LOYALTY has drop {}
The init
function of the module uses the LOYALTY
OTW to create the token. All init
functions run one time only at the package publish event. The initializer function makes use of the OTW LOYALTY
type defined previously in its call to create_currency
. The function also defines a policy, sending both the policy capability and trasury capability to the address associated with the publish event. The holder of these transferrable capabilities can mint new LOYALTY
tokens and modify their policies.
fun init(otw: LOYALTY, ctx: &mut TxContext) {
let (treasury_cap, coin_metadata) = coin::create_currency(
otw,
0, // no decimals
b"LOY", // symbol
b"Loyalty Token", // name
b"Token for Loyalty", // description
option::none(), // url
ctx
);
let (mut policy, policy_cap) = token::new_policy(&treasury_cap, ctx);
token::add_rule_for_action<LOYALTY, GiftShop>(
&mut policy,
&policy_cap,
token::spend_action(),
ctx
);
token::share_policy(policy);
transfer::public_freeze_object(coin_metadata);
transfer::public_transfer(policy_cap, tx_context::sender(ctx));
transfer::public_transfer(treasury_cap, tx_context::sender(ctx));
}
The LOYALTY
minting function is called reward_user
. As mentioned previously, the holder of the TreasuryCap
can call this function to mint new loyalty tokens and send them to the desired address. The function uses the token::mint
function to create the token and token::transfer
to send it to the intended recipient.
public fun reward_user(
cap: &mut TreasuryCap<LOYALTY>,
amount: u64,
recipient: address,
ctx: &mut TxContext
) {
let token = token::mint(cap, amount, ctx);
let req = token::transfer(token, recipient, ctx);
token::confirm_with_treasury_cap(cap, req, ctx);
}
Finally, the example includes a buy_a_gift
function to handle the redemption of LOYALTY
tokens for Gift
types. The function ensures the gift price matches the number of loyalty tokens spent, then uses the token::spend
function to handle the treasury bookkeeping.
public fun buy_a_gift(
token: Token<LOYALTY>,
ctx: &mut TxContext
): (Gift, ActionRequest<LOYALTY>) {
assert!(token::value(&token) == GIFT_PRICE, EIncorrectAmount);
let gift = Gift { id: object::new(ctx) };
let mut req = token::spend(token, ctx);
token::add_approval(GiftShop {}, &mut req, ctx);
(gift, req)
}
Complete code
Toggle display of the complete source for this example, including comments, or use the link in the Related links section to view the project source on GitHub.
loyalty.move
loyalty.move
Related links
- Closed Loop Token standard: Details for the standard used to create tokens on Sui.
- Source code: The source code in GitHub for this example.
- In-Game Tokens: Example of how to create tokens for use as in-game currency.
- Regulated Coin and Deny List: Example of how to create regulated coins on the Sui network.