Files
pwap/shared/lib/wallet.ts
T
pezkuwichain 06a7ec9424 feat(frontend): align wUSDT Asset ID with SDK (1000)
This commit aligns the frontend wUSDT implementation with the SDK runtime
constants, ensuring consistency across the entire stack.

Changes:
- Update ASSET_IDS.WUSDT from 2 → 1000 (matches SDK constants)
- Add ASSET_CONFIGS with wUSDT configuration (6 decimals, min balance)
- Update all asset queries in AccountBalance.tsx to use ASSET_IDS.WUSDT
- Update pool queries for wHEZ/wUSDT and PEZ/wUSDT pools
- Update USDTBridge.tsx burn transaction to use ASSET_IDS.WUSDT
- Refactor usdt.ts to reference ASSET_CONFIGS instead of hardcoded values

Asset ID Allocation Strategy:
- 0-999: Reserved for protocol tokens (wHEZ, PEZ, etc.)
- 1000+: Bridged/wrapped external assets (wUSDT, etc.)

Technical Details:
- wUSDT uses 6 decimals (USDT standard), not 12 like native HEZ
- All frontend code now uses centralized ASSET_CONFIGS
- ESLint passes with 0 errors (8 pre-existing warnings unrelated to changes)

This is part of Phase 1 wUSDT infrastructure setup, working in parallel
with SDK benchmarking builds currently running in background.

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-20 19:52:48 +03:00

171 lines
6.1 KiB
TypeScript

// ========================================
// PezkuwiChain - Substrate/Polkadot.js Configuration
// ========================================
// This file configures wallet connectivity for Substrate-based chains
import type { InjectedAccountWithMeta } from '@polkadot/extension-inject/types';
// ========================================
// NETWORK ENDPOINTS
// ========================================
export const NETWORK_ENDPOINTS = {
local: import.meta.env.VITE_DEVELOPMENT_WS || 'ws://127.0.0.1:9944',
testnet: import.meta.env.VITE_TESTNET_WS || 'wss://testnet.pezkuwichain.io',
mainnet: import.meta.env.VITE_MAINNET_WS || 'wss://mainnet.pezkuwichain.io',
staging: import.meta.env.VITE_STAGING_WS || 'wss://staging.pezkuwichain.io',
beta: import.meta.env.VITE_BETA_WS || 'wss://beta.pezkuwichain.io',
};
// ========================================
// CHAIN CONFIGURATION
// ========================================
export const CHAIN_CONFIG = {
name: import.meta.env.VITE_CHAIN_NAME || 'PezkuwiChain',
symbol: import.meta.env.VITE_CHAIN_TOKEN_SYMBOL || 'PEZ',
decimals: parseInt(import.meta.env.VITE_CHAIN_TOKEN_DECIMALS || '12'),
ss58Format: parseInt(import.meta.env.VITE_CHAIN_SS58_FORMAT || '42'),
};
// ========================================
// SUBSTRATE ASSET IDs (Assets Pallet)
// ========================================
// ⚠️ IMPORTANT: HEZ is the native token and does NOT have an Asset ID
// Only wrapped/asset tokens are listed here
//
// Asset ID Allocation:
// - 0-999: Reserved for protocol tokens (wHEZ, PEZ, etc.)
// - 1000+: Bridged/wrapped external assets (wUSDT, etc.)
export const ASSET_IDS = {
WHEZ: parseInt(import.meta.env.VITE_ASSET_WHEZ || '0'), // Wrapped HEZ
PEZ: parseInt(import.meta.env.VITE_ASSET_PEZ || '1'), // PEZ utility token
WUSDT: parseInt(import.meta.env.VITE_ASSET_WUSDT || '1000'), // Wrapped USDT (6 decimals, matches SDK)
USDT: parseInt(import.meta.env.VITE_ASSET_USDT || '3'),
BTC: parseInt(import.meta.env.VITE_ASSET_BTC || '4'),
ETH: parseInt(import.meta.env.VITE_ASSET_ETH || '5'),
DOT: parseInt(import.meta.env.VITE_ASSET_DOT || '6'),
} as const;
// ========================================
// ASSET CONFIGURATIONS
// ========================================
export const ASSET_CONFIGS = {
WHEZ: {
id: ASSET_IDS.WHEZ,
symbol: 'wHEZ',
name: 'Wrapped HEZ',
decimals: 12,
minBalance: 1_000_000_000, // 0.001 HEZ
},
PEZ: {
id: ASSET_IDS.PEZ,
symbol: 'PEZ',
name: 'PEZ Utility Token',
decimals: 12,
minBalance: 1_000_000_000, // 0.001 PEZ
},
WUSDT: {
id: ASSET_IDS.WUSDT,
symbol: 'wUSDT',
name: 'Wrapped USDT',
decimals: 6, // ⚠️ CRITICAL: wUSDT uses 6 decimals (USDT standard), not 12!
minBalance: 1_000, // 0.001 USDT
},
} as const;
// ========================================
// EXPLORER URLS
// ========================================
export const EXPLORER_URLS = {
polkadotJs: import.meta.env.VITE_EXPLORER_URL || 'https://polkadot.js.org/apps/?rpc=',
custom: import.meta.env.VITE_CUSTOM_EXPLORER_URL || 'https://explorer.pezkuwichain.io',
};
// ========================================
// WALLET ERROR MESSAGES
// ========================================
export const WALLET_ERRORS = {
NO_EXTENSION: 'No Polkadot.js extension detected. Please install Polkadot.js or compatible wallet.',
NO_ACCOUNTS: 'No accounts found. Please create an account in your wallet extension.',
CONNECTION_FAILED: 'Failed to connect wallet. Please try again.',
TRANSACTION_FAILED: 'Transaction failed. Please check your balance and try again.',
USER_REJECTED: 'User rejected the request.',
INSUFFICIENT_BALANCE: 'Insufficient balance to complete transaction.',
INVALID_ADDRESS: 'Invalid address format.',
API_NOT_READY: 'Blockchain API not ready. Please wait...',
};
// ========================================
// UTILITY FUNCTIONS
// ========================================
/**
* Format Substrate address for display (SS58 format)
* @param address - Full substrate address
* @returns Shortened address string (e.g., "5GrwV...xQjz")
*/
export const formatAddress = (address: string): string => {
if (!address) return '';
return `${address.slice(0, 6)}...${address.slice(-4)}`;
};
/**
* Format balance from planck to human-readable format
* @param balance - Balance in smallest unit (planck)
* @param decimals - Token decimals (default 12 for PEZ)
* @returns Formatted balance string
*/
export const formatBalance = (balance: string | number, decimals = 12): string => {
if (!balance) return '0';
const value = typeof balance === 'string' ? parseFloat(balance) : balance;
return (value / Math.pow(10, decimals)).toFixed(4);
};
/**
* Parse human-readable amount to planck (smallest unit)
* @param amount - Human-readable amount
* @param decimals - Token decimals
* @returns Amount in planck
*/
export const parseAmount = (amount: string | number, decimals = 12): bigint => {
const value = typeof amount === 'string' ? parseFloat(amount) : amount;
return BigInt(Math.floor(value * Math.pow(10, decimals)));
};
/**
* Get asset symbol by ID
* @param assetId - Asset ID from Assets pallet
* @returns Asset symbol or 'UNKNOWN'
*/
export const getAssetSymbol = (assetId: number): string => {
const entry = Object.entries(ASSET_IDS).find(([_, id]) => id === assetId);
return entry ? entry[0] : 'UNKNOWN';
};
/**
* Get current network endpoint based on VITE_NETWORK env
* @returns WebSocket endpoint URL
*/
export const getCurrentEndpoint = (): string => {
const network = import.meta.env.VITE_NETWORK || 'local';
return NETWORK_ENDPOINTS[network as keyof typeof NETWORK_ENDPOINTS] || NETWORK_ENDPOINTS.local;
};
// ========================================
// TYPE DEFINITIONS
// ========================================
export interface PolkadotWalletState {
isConnected: boolean;
accounts: InjectedAccountWithMeta[];
selectedAccount: InjectedAccountWithMeta | null;
balance: string;
error: string | null;
}
export const initialPolkadotWalletState: PolkadotWalletState = {
isConnected: false,
accounts: [],
selectedAccount: null,
balance: '0',
error: null,
};