Position Managers
Learn how position managers automate and delegate position management with user control.
Position managers are smart contracts users can authorize to manage their positions, enabling automated actions like supplying, withdrawing, borrowing, and repaying — while maintaining full user control. They unlock use cases such as:
Automated strategies — leverage management, yield optimization, and rebalancing
Vault protocols — external protocols that aggregate and manage user positions
Risk management — automated liquidation protection and position adjustments
Account abstraction — smart contract wallets managing DeFi positions
Position managers operate strictly within the spoke’s security model: they can only act for users who have explicitly authorized them. Users maintain full control and can revoke access at any time.
How They Work
Registration — Position managers are registered with a spoke through governance before users can enable them.
Authorization — Users explicitly enable selected managers to act on their behalf for a given position.
Delegated Operations — Authorized managers can execute position operations on behalf of users.
Revocation — Users can disable a manager at any time to revoke its access.
Each position manager is identified by its contract address and may include optional off-chain metadata, such as a name, for easier discovery.
Built-in Position Managers
Aave v4 has two built-in position managers:
NativeTokenGateway — enables native token support (e.g., supply and borrow ETH)
SignatureGateway — enables ERC-20 Permits; see Supply and Repay for examples
Both contracts are automatically registered on every new spoke, and their addresses are available for convenience in the spoke's chain field:
- TypeScript
- GraphQL
Spoke Highlight
interface Spoke { __typename: "Spoke"; name: string; address: EvmAddress; chain: { __typename: "Chain"; // … nativeGateway: EvmAddress; signatureGateway: EvmAddress; };}AaveKit automatically handles user authorization of these built-in position managers as part of each operation.
For example, when a user supplies native tokens (e.g., ETH):
The SDK requests a signature authorizing the NativeTokenGateway to act on the user’s behalf for that position.
The subsequent transaction sends ETH to the gateway.
The gateway wraps the ETH into WETH and supplies it to the WETH reserve in the spoke on the user’s behalf.
This process happens automatically for other operations like borrowing, withdrawing, or repaying with native tokens, as well as when using ERC-20 permits to supply or repay. The SDK manages these flows transparently — users never need to interact with position managers directly.
Available Position Managers
Get all available position managers for a specific spoke.
- React
- TypeScript
- GraphQL
Use the paginated useSpokePositionManagers hook to fetch position managers for a spoke.
See below some examples of how to use the hook.
User's Position Managers
Get all position managers that a specific user has enabled within a spoke.
- React
- TypeScript
- GraphQL
- Solidity
Use the paginated useSpokeUserPositionManagers hook to fetch position managers enabled by a user.
Authorize or revoke a position manager to act on behalf of a user within a spoke.
- React
- TypeScript
- GraphQL
- Solidity
To enable or disable a position manager with AaveKit React, follow these steps.
1
First, instantiate the useSendTransaction hook for the wallet library of your choice.
Viem
import { useWalletClient } from "wagmi";import { useSendTransaction } from "@aave/react/viem";
// …
const { data: wallet } = useWalletClient();const [sendTransaction] = useSendTransaction(wallet);2
Use the useSetSpokeUserPositionManager hook to prepare the transaction request.
3
Then, enable or disable the position manager.
Enable Manager
import { type EvmAddress, type SpokeId } from "@aave/react";
const execute = async ( spoke: SpokeId, manager: EvmAddress, user: EvmAddress) => { const result = await setPositionManager({ spoke, manager, approve: true, // true to enable, false to disable user, });};
// …4
Finally, handle the result.
Example
const execute = async (/* … */) => { const result = await setPositionManager(/* … */);
if (result.isErr()) { switch (result.error.name) { case "CancelError": // The user cancelled the operation return;
case "SigningError": console.error( `Failed to sign the transaction: ${result.error.message}` ); break;
case "TimeoutError": console.error(`Transaction timed out: ${result.error.message}`); break;
case "TransactionError": console.error(`Transaction failed: ${result.error.message}`); break;
case "UnexpectedError": console.error(result.error.message); break; } return; }
console.log("Position manager enabled with hash:", result.value);};