Publishing, Deploying & Local Network
MCP tool: When available in your environment, also query the Sui documentation MCP server (
) for up-to-date answers. Use it for verification and for details not covered by these reference files.
Source constraint: All information sourced exclusively from
docs.sui.io and
MystenLabs/sui-stack-hello-world.
Publishing a package
Canonical hello-world publish flow
For the full-stack starter, publish the existing hello-world package only:
bash
cd sui-stack-hello-world/move/hello-world
sui move build
sui client publish
Use the package ID from the publish output to update
sui-stack-hello-world/ui/src/constants.ts
(
TESTNET_HELLO_WORLD_PACKAGE_ID
). Do not publish a separate counter package, and do not create a second project directory.
Pre-publish checklist
- Verify your active environment:
- Verify you have SUI tokens:
- Build successfully:
Publish
This deploys the package to the active network and returns:
- A unique package ID (use this for all future interactions)
- An UpgradeCap object (sent to your address, controls future upgrades)
- Object IDs for anything created during functions
Test publishing (ephemeral networks)
Use
to publish a package to an ephemeral environment for testing without persisting state to a real network:
This publishes the package, runs
functions, and returns the same output as
(package ID, UpgradeCap, created objects), but the deployment is not permanent. Use it to:
- Verify that functions execute correctly before committing to a real publish
- Test publish + upgrade flows in CI without consuming Testnet/Devnet resources
- Validate gas costs and object creation before a Mainnet deploy
respects
for multi-environment packages:
bash
sui client test-publish --build-env testnet
After publishing
The publish transaction output lists the package ID under the
created objects section (alongside the
and any objects created by
functions). The
field is also automatically added to your
. To interact with the published package:
bash
# Call a function
sui client call --package <PACKAGE_ID> --module greeting --function new
# Query an object
sui client object <OBJECT_ID>
"Your package is already published" error
If you see this error when running
, it means
already has an entry for your active environment. This happens when iterating on a package during development.
- To publish to a different network, switch environments with
sui client switch --env <ENV>
and run again. The toolchain tracks published addresses per environment in automatically — do not delete .
- To upgrade the existing package on the same network, use (see below).
Upgrading a published package
Published packages are immutable, but you can upgrade by publishing a new version linked to the original. The
object controls upgrade authority.
Important: you can restrict the
in the same PTB as the
command — for example, publishing and immediately calling
sui::package::only_additive_upgrades
in one atomic transaction. You can also destroy it entirely to make the package permanently immutable (see upgrade policies below).
bash
sui client upgrade --upgrade-capability <CAP_ID>
Finding your UpgradeCap
The
object ID is needed for every upgrade. There are several ways to find it:
- Published.toml (preferred): After publishing, the toolchain records the cap ID in under the field for each environment.
- Query owned objects: List all objects owned by the publish address:
bash
sui client objects --type 0x2::package::UpgradeCap
- Publish transaction output: The original output includes the object ID in the created objects list.
- Explorer: Search for your address on SuiVision () or Suiscan () and filter owned objects by type .
Upgrade policies
Upgrade policies restrict what can change:
- Compatible (default): The most permissive policy. See detailed rules below.
- Additive: New modules can be added, but existing modules cannot change at all.
- Dependency-only: Only dependency versions can be updated. No code changes.
Restricting the UpgradeCap in the same PTB as publish: You can restrict the
in the same programmable transaction block as the
command itself — for example, calling
sui::package::only_additive_upgrades
on the
immediately after publishing, all within a single atomic transaction. This is the recommended approach for locking down upgrade policy from the start. Once restricted, you cannot widen the policy.
Other UpgradeCap options:
- Transfer to a multisig address for shared upgrade governance.
- Destroy the UpgradeCap to make the package permanently immutable. Call
sui::package::make_immutable
, which consumes and destroys the object. Once the cap is destroyed, no one can ever upgrade the package again — this is irreversible.
Compatible upgrade rules (detailed)
Under the compatible policy, these changes are allowed:
- Add new functions (public or private)
- Add new modules
- Change function implementations (body)
- Add new struct types
- Change private/friend function signatures
These changes break compatibility and will be rejected:
- Remove or rename an existing module
- Remove or rename a public function
- Change a public function's signature (parameters, return types, type parameters)
- Remove, rename, or reorder struct fields
- Change the type of a struct field
- Add or remove struct abilities (, , , )
- Remove a struct type entirely
- Change a struct's type parameters
Before upgrading, review your diff against these rules. The
command will reject incompatible changes at build time with a descriptive error.
Type anchoring after upgrades
Struct types are permanently anchored to the original package ID where they were first published. After an upgrade, the new package gets a new ID, but all objects created by the upgraded code still have their type rooted in the original package ID.
This has critical implications:
- Querying objects by type (e.g., with a filter) must use the original package ID.
- Calling functions via must use the upgraded (latest) package ID.
- Frontend apps should maintain both IDs: for type queries and for function calls.
ts
// Original publish → package ID 0x1234...
// After upgrade → package ID 0x5678...
// Query: use ORIGINAL package ID
client.core.listOwnedObjects({
owner: addr,
type: '0x1234...::module::MyObject', // ✅ original ID
});
// Call: use UPGRADED package ID
tx.moveCall({
target: '0x5678...::module::my_function', // ✅ upgraded ID
});
Publishing to multiple networks
To publish to a different network (for example, from Testnet to Devnet), switch environments and publish again. Each network gives the package a different ID. The
file tracks published addresses per environment.
Before publishing to a new network, ensure you have tokens for that network:
- Testnet: Free tokens through the web faucet at , Discord ( in ), or the TypeScript SDK (). does not work on Testnet.
- Devnet: Free tokens via , the web faucet at , Discord ( in ), or the TypeScript SDK.
- Localnet: Free tokens via or the local faucet at or (started with
sui start --with-faucet --force-regenesis
).
- Mainnet: SUI tokens with real monetary value. Acquire through exchanges or transfers. No faucet available.
Serializing for external signing
To generate transaction bytes for signing by another party (for example, a multisig):
bash
sui client publish --serialize-output
This outputs base64 transaction bytes instead of executing.
Local network (localnet)
Localnet runs a full Sui network on your machine for offline development and rapid iteration. Start it with:
bash
sui start --with-faucet --force-regenesis
The
flag resets all on-chain state each time the network starts, giving you a clean environment on every restart. The
flag starts a local faucet so you can fund addresses.
To connect the CLI to your localnet:
bash
sui client switch --env localnet
Get local tokens via
or by hitting the local faucet endpoint directly at
or
.
Localnet is useful for:
- Offline development without depending on Testnet/Devnet availability
- Rapid iteration on publish and upgrade flows (reset state with each restart)
- Testing functions and object creation before deploying to a shared network
Mainnet launch checklist
Use this checklist when preparing a package for Mainnet publishing. Every item should be verified before executing the publish transaction.
1. Tests and coverage
Run the full test suite and confirm all tests pass:
For coverage reporting (if your project requires a threshold):
bash
sui move test --coverage
sui move coverage summary
Fix any failing tests before proceeding. Do not publish untested code to Mainnet.
2. Dependencies and addresses
- Verify uses and has no legacy section or git-based Sui framework dependency.
- Confirm includes a entry with the correct chain ID.
- If using MVR dependencies (
{ r.mvr = "@org/package" }
), verify they resolve on Mainnet.
- Run to confirm clean compilation with no warnings.
3. Upgrade policy decision
Decide your upgrade policy before publishing — you cannot widen it later:
| Policy | What you can change | When to use |
|---|
| Compatible (default) | Add functions, add modules, update implementations. Cannot remove functions or change struct layouts. | Most packages — gives flexibility for bug fixes while preserving type safety. |
| Additive | Add new modules only. Existing modules are frozen. | Packages where you want to extend functionality but guarantee existing code never changes. |
| Dependency-only | Only update dependency versions. | Nearly-finalized packages that should only track framework updates. |
| Immutable | Nothing. Package is permanently frozen. | Fully audited packages where immutability is a trust guarantee (e.g., token contracts). |
To restrict the policy in the same transaction as publish, include a
to
sui::package::only_additive_upgrades
,
, or
on the
in your publish PTB.
4. Gas estimation
Mainnet SUI has real monetary value. Estimate gas before publishing:
bash
sui client publish --dry-run
The dry-run output includes
,
, and
. The total gas required is
computationCost + storageCost - storageRebate
. Ensure your address holds enough SUI to cover this amount plus a margin.
5. Signer and custody plan
Decide who controls the publish address and the
:
- Single signer: Simplest. One key publishes and holds the . Suitable for personal projects or early-stage development.
- Multisig: For teams or high-value packages. Create a multisig address, publish using , and have the required signers sign offline. Transfer the to the multisig address in the same PTB as publish.
- Immutable on publish: If no upgrades will ever be needed, destroy the in the publish PTB (
sui::package::make_immutable
). This removes custody concerns entirely.
For multisig publishing:
bash
# Generate unsigned transaction bytes
sui client publish --serialize-output
# Each signer signs the bytes, then combine and execute
6. Final pre-publish verification
Before executing the publish transaction on Mainnet:
Dry runs and transaction debugging
A dry run simulates a transaction without submitting it to the network. Use dry runs to:
- Estimate gas costs before execution.
- Verify that a transaction succeeds before asking a user to sign.
- Debug failing transactions by inspecting the error before spending gas.
Wallets (like Slush) automatically perform dry runs before presenting a transaction for signing. If a dry run fails, the wallet shows an error instead of prompting.
From the TypeScript SDK, use
devInspectTransactionBlock
to dry-run a transaction programmatically. From the CLI, the
flag simulates execution.
When debugging a dry run failure: check that all object IDs are correct, the object versions are current, the sender has sufficient gas, the function arguments match the expected types, and the active environment (
) matches the network where the package is published.
Production monitoring
Sui packages are immutable once published, so monitoring is critical — you cannot hotfix a live contract, only publish an upgrade.
What to monitor
| Signal | How | Why |
|---|
| Failed transactions involving your package | Subscribe to transaction effects via gRPC streaming, filter by package ID | Detects Move aborts, gas failures, or unexpected reverts in production |
| Gas spend | Track from transaction effects | Catch unexpectedly expensive operations or gas drain attacks |
| Event emission | Subscribe to events by type ({packageId}::module::EventName
) via gRPC streaming | Core business telemetry — mints, transfers, admin actions, deny list changes |
| Object creation/deletion rates | Query or subscribe to object changes filtered by your types | Detect abnormal activity (mass minting, object spam) |
| Admin/cap usage | Filter events for capability-gated actions | Detect unauthorized or unexpected admin operations |
| Shared object contention | Monitor transaction latency for shared-object transactions | High contention degrades UX; may need object sharding |
Implementation
Use gRPC streaming subscriptions for real-time monitoring:
ts
for await (const event of client.subscriptionService.subscribeEvents({
filter: { MoveEventModule: { package: PACKAGE_ID, module: 'my_module' } },
})) {
// Forward to your monitoring stack (Grafana, Datadog, PagerDuty, etc.)
}
For historical analysis, run a custom indexer (
) that writes relevant events and transaction effects to your own database. See the
skill's
.
Emit events for every security-critical action in your Move code — admin changes, configuration updates, deny list modifications, object deletions. Events are the only way offchain systems can observe these actions.
Rollback and incident response
Sui packages cannot be rolled back. Published bytecode is immutable. There is no
or
command. Recovery means publishing a forward-fix upgrade.
If a bad upgrade is published
- Assess scope. Determine which functions are affected. Existing objects created by prior versions are still valid — their types are anchored to the original package ID.
- Publish a fix upgrade immediately. Write the corrected code, run tests, dry-run on Testnet, then on Mainnet. The new package ID replaces the old one for all future calls.
- Update frontends. Point to the new (fixed) package ID. Type queries still use .
- Communicate. If the bug affected user-facing behavior, notify users through your app's channels.
If the UpgradeCap is compromised
An attacker with the
can publish arbitrary code under your package. Mitigation:
- If you still hold the cap: Immediately restrict it ( or ) to prevent further malicious upgrades.
- If the attacker holds the cap: You cannot recover upgrade authority. Publish a new package, migrate users, and communicate the migration. This is why multisig custody of the matters for production packages.
If a shared object is corrupted
A buggy function may write invalid state to a shared object. Since shared objects are mutable by any transaction:
- If you can upgrade: Publish an upgrade with a repair function that fixes the corrupted state. Gate it behind an .
- If the package is immutable: The only option is to deploy a new package with a migration function that reads the old object's data (if accessible) and creates corrected objects.
Prevention checklist