@effectstream/wallets
Package:
@effectstream/wallets· Source
Browser wallet connectors and the runtime client an EffectStream frontend uses to log in, sign batcher messages, send transactions, and wait for them to be processed. Spans MetaMask (and any EVM-injected wallet), Cardano (CIP-30), Midnight, Mina, Polkadot, Algorand, and Avail.
- Browser wallet connectors plus the runtime client a frontend uses to log in, sign, and submit.
- Spans MetaMask + injected EVM, Cardano (CIP-30), Midnight, Mina, Polkadot, Algorand, Avail.
- Used together with
@effectstream/crypto(server-side verification). - High-level helpers
walletLoginandsendTransactionhide the polling loop. - Also supports local (in-browser) wallets that need no extension:
CardanoLocal,EvmViem,MidnightLocal.
Install
bun add @effectstream/wallets
# or
npm install @effectstream/wallets
Standalone usage
The high-level path most apps use: log in with a wallet, then send a transaction through the batcher and wait until it lands in a block.
import {
walletLogin,
WalletMode,
EffectstreamConfig,
sendTransaction,
} from "@effectstream/wallets";
const config = new EffectstreamConfig({ /* …chain/batcher URLs… */ });
const wallet = await walletLogin({
config,
preference: { name: "MetaMask" },
mode: WalletMode.EvmInjected,
});
const result = await sendTransaction(wallet, "join|alice");
// result.blockHeight === number once the batcher's submission landed
If you just want to discover available wallets:
import { allInjectedWallets, WalletMode } from "@effectstream/wallets";
const available = await allInjectedWallets({
signatureSupport: true,
transactionSupport: true,
});
const evmOptions = available[WalletMode.EvmInjected]; // [{ metadata, ... }, …]
Browser only. This package depends on
window.ethereum, the Cardano CIP-30 API, etc. - it won't load in plain Node. Server-side signature verification is@effectstream/crypto.
Local wallets (no extension required)
Besides connecting an injected/extension wallet, @effectstream/wallets
can generate and manage a key-pair locally, in the browser. The local
modes return the same Wallet handle as the injected ones, so signMessage
and sendTransaction work identically:
WalletMode.CardanoLocal- BIP-39 seed generated in-browser (via Lucid).WalletMode.EvmViem- viem account from a0xprivate key.WalletMode.MidnightLocal- hex seed generated in-browser.
import { walletLogin, WalletMode, signMessage } from "@effectstream/wallets";
// A fresh BIP-39 seed is generated in the browser when `seedPhrase` is omitted.
const wallet = await walletLogin({
mode: WalletMode.CardanoLocal,
network: "Preview", // "Mainnet" | "Preprod" | "Preview" | "Custom"
});
if (!wallet.success) throw new Error(wallet.errorMessage);
wallet.result.walletAddress; // addr_test1...
const signature = await signMessage(wallet.result, "hello effectstream");
These modes need no browser extension (no window.ethereum, no CIP-30) -
the key material is created client-side and never touches a server. Pair a
local key with the account-linking delegation (&linkAddress) so a real
wallet (Cardano CIP-30, MetaMask, ...) can authorise it to sign
non-financial inputs: this is the basis of the device-key / frictionless
signing flow. Source: src/cardano/local.ts
and the runnable src/cardano/local.test.ts
(generate -> sign -> verify with @effectstream/crypto).
Inside EffectStream
The client-side counterpart to @effectstream/crypto. Frontends
connect a wallet, sign a batcher login or transaction, and POST it to
the batcher HTTP endpoint (@effectstream/batcher-sdk). The node's
state machine then verifies the signature with
CryptoManager.getCryptoManager(addressType). The runtime helpers
exported here (sendTransaction, waitForEffectstreamBlockProcessed)
hide the polling loop.
Key exports
Login + signing:
walletLogin(args): top-level helper that discovers, connects, and produces a signed batcher login.EffectstreamConfig: runtime config (chain URLs, batcher URL, ...) the helpers consume.signMessage(wallet, message)- sign an arbitrary message with the connected wallet.
Sending transactions:
sendTransaction(wallet, conciseInput, opts?): submit through the batcher and (by default) wait for processing.sendBatcherTransaction(...): explicit batcher submission.sendSelfSequencedTransaction(...)bypasses the batcher.waitForEffectstreamBlockProcessed(...)- block-height poller used by the above.
Wallet discovery / identification:
WalletMode: enum ofEvmInjected,EvmEthers,EvmViem,Cardano,CardanoLocal,Midnight,MidnightLocal,Polkadot,Algorand,Mina,AvailJs. The*LocalandEvmViemmodes are in-browser wallets that need no extension.WalletNameMap:Record<WalletMode, string>for display.allInjectedWallets(config)lists installed wallets in the browser.getAddressType(walletMode)maps aWalletModeto its@effectstream/utilsAddressType.
Types:
Wallet: handle returned bywalletLogin, used by send/sign helpers.UserSignature,Hash,BatcherPostResponse,BatcherTrackResponse,PostDataResponse,PostDataResponseAsync,SignFunction.
Lower-level connector machinery (used internally; export surface is
stable but rarely needed in app code): connectInjectedWallet,
WalletModeMap, IProvider, IConnector, IInjectedConnector,
InjectionPreference.
Examples
Runnable: src/utils.test.ts and
test/examples.test.ts.
Real-world usage in templates (each imports walletLogin, WalletMode,
EffectstreamConfig, sendTransaction):