بنقرة واحدة
move-syntax
// Move language syntax — module layout, imports, mutability, visibility, method syntax, enums, macros, and comments.
// Move language syntax — module layout, imports, mutability, visibility, method syntax, enums, macros, and comments.
| name | move-syntax |
| description | Move language syntax — module layout, imports, mutability, visibility, method syntax, enums, macros, and comments. |
Use the new single-line module declaration without braces:
// ✅
module my_package::my_module;
// ❌ Legacy — do not use
module my_package::my_module {
...
}
Standard section order within a module:
use importsconst)fun init (if needed)public(package) functions#[test_only])Use === Section Title === comments to delimit sections for readability.
use import rulesDon't use a lone {Self} — just import the module directly:
// ✅
use my_package::my_module;
// ❌ Redundant braces
use my_package::my_module::{Self};
When importing both the module and members, group them with Self:
// ✅
use sui::coin::{Self, Coin};
// ❌ Separate imports
use sui::coin;
use sui::coin::Coin;
mut is RequiredAll variables that are reassigned or mutably borrowed must be declared let mut:
// ✅
let mut pool = Pool { id: object::new(ctx), ... };
let mut balance = balance::zero<SUI>();
// ❌ Legacy
let pool = Pool { id: object::new(ctx), ... };
Function parameters that are mutably borrowed must also be mut:
public fun deposit(mut pool: Pool, coin: Coin<SUI>): Pool { ... }
| Keyword | Scope |
|---|---|
public | Callable from any module |
public(package) | Callable only within the same package |
| (none) | Private — callable only within the same module |
public(friend) and friend declarations are deprecated. Use public(package) instead.
// ✅
public(package) fun internal_logic(pool: &mut Pool) { ... }
// ❌ Deprecated
friend my_package::other_module;
public(friend) fun internal_logic(pool: &mut Pool) { ... }
Never use public entry — use one or the other. public functions are composable in PTBs and return values; entry functions are transaction endpoints only and cannot return values:
// ✅ Composable — can be chained in PTBs
public fun mint(ctx: &mut TxContext): NFT { ... }
// ✅ Transaction endpoint — no return value needed
entry fun mint_and_transfer(recipient: address, ctx: &mut TxContext) { ... }
// ❌ Redundant combination — avoid
public entry fun mint(ctx: &mut TxContext): NFT { ... }
Always order parameters: mutable objects → immutable objects → capabilities → primitive types → &Clock → &mut TxContext:
// ✅
public fun call(
app: &mut App,
config: &Config,
cap: &AdminCap,
amount: u64,
is_active: bool,
clock: &Clock,
ctx: &mut TxContext,
) { }
// ❌ Wrong order
public fun call(
amount: u64,
app: &mut App,
cap: &AdminCap,
config: &Config,
ctx: &mut TxContext,
) { }
Name getters after the field. Do not use a get_ prefix:
// ✅
public fun fee_bps(pool: &Pool): u64 { pool.fee_bps }
public fun fee_bps_mut(pool: &mut Pool): &mut u64 { &mut pool.fee_bps }
// ❌
public fun get_fee_bps(pool: &Pool): u64 { pool.fee_bps }
Functions whose first argument matches a type are automatically callable as methods:
// Given:
public fun value(coin: &Coin<SUI>): u64 { coin.value() }
// Both are valid, prefer the method form:
let v = coin.value(); // ✅ method syntax — prefer this
let v = coin::value(&coin); // ✅ also valid, but more verbose
Use method syntax wherever it improves readability. Declare use fun aliases for functions defined outside the owning module:
use fun my_module::pool_value as Pool.value;
Use enums for types with multiple variants. Enums cannot have the key ability (they cannot be top-level objects), but they can be stored inside objects:
public enum OrderStatus has copy, drop, store {
Pending,
Filled { amount: u64 },
Cancelled,
}
Pattern match with match:
match (order.status) {
OrderStatus::Pending => { ... },
OrderStatus::Filled { amount } => { /* use amount */ },
OrderStatus::Cancelled => { ... },
}
Use macro functions for higher-order patterns instead of manual loops.
// Do something N times
32u8.do!(|_| do_action());
// Build a new vector from an index range
let v = vector::tabulate!(32, |i| i);
// Iterate by immutable reference
vec.do_ref!(|e| process(e));
// Iterate by mutable reference
vec.do_mut!(|e| *e = *e + 1);
// Consume vector, calling a function on each element
vec.destroy!(|e| handle(e));
// Fold into a single value
let sum = vec.fold!(0u64, |acc, x| acc + x);
// Filter (requires T: drop)
let big = vec.filter!(|x| *x > 100);
All of these replace verbose manual while loops. Use them whenever you iterate over a vector.
// Execute a function if Some, then drop
opt.do!(|value| process(value));
// Unwrap with a default (or abort)
let value = opt.destroy_or!(default);
let value = opt.destroy_or!(abort ECannotBeEmpty);
These replace verbose if (opt.is_some()) / destroy_some() patterns:
// ❌ Verbose
if (opt.is_some()) {
let inner = opt.destroy_some();
process(inner);
};
// ✅
opt.do!(|inner| process(inner));
Use /// for doc comments. JavaDoc-style /** */ is not supported in Move:
/// Returns the current fee in basis points.
public fun fee_bps(pool: &Pool): u64 { pool.fee_bps }
// ❌ Not supported
/** Returns the current fee in basis points. */
public fun fee_bps(pool: &Pool): u64 { pool.fee_bps }
Use regular // comments to explain non-obvious logic, potential edge cases, and TODOs:
// Note: can underflow if reserve is smaller than minimum_liquidity.
// TODO: add assert! guard before production use.
let lp_supply = math::sqrt(reserve_x * reserve_y);
Full-stack Sui blockchain development — Move smart contracts, TypeScript SDK, and frontend dApp Kit. Routes to the appropriate sub-skill based on what the user is building.
Sui object model — struct declarations, abilities (key/store/copy/drop), object ownership, naming conventions, and dynamic fields.
Move design patterns — events, error handling, one-time witness (OTW), capability pattern, and pure functions/composability.
Move package setup (Move.toml, edition, dependencies), building, testing, and common pitfalls from other Move dialects.
Common Sui Move standard library patterns — strings, Coin/Balance, Option, addresses, UID, TxContext, vectors, and struct unpacking.
Sui frontend dApp development with @mysten/dapp-kit-react (React) and @mysten/dapp-kit-core (Vue, vanilla JS, other frameworks). Use when building browser apps that connect to Sui wallets, query on-chain data, or execute transactions. Use alongside the sui-ts-sdk skill for PTB construction patterns.