claiming uifee

This commit is contained in:
2025-06-08 22:34:30 +07:00
parent 350cab44eb
commit 8836c45c9d
2 changed files with 194 additions and 0 deletions

View File

@@ -140,6 +140,7 @@ declare module 'fastify' {
claimGmxFundingFees: typeof claimGmxFundingFees;
claimGmxPriceImpact: typeof claimGmxPriceImpact;
getGmxPriceImpactRebates: typeof getGmxPriceImpactRebates;
claimGmxUiFees: typeof claimGmxUiFees;
}
}
@@ -175,6 +176,11 @@ const getPriceImpactRebatesSchema = z.object({
account: z.string().nonempty()
});
// Schema for claim UI fees request
const claimUiFeesSchema = z.object({
account: z.string().nonempty()
});
/**
* Gets a GMX SDK client initialized for the given address
* If a walletId is provided, it will be used with Privy for signing
@@ -920,6 +926,7 @@ export default fp(async (fastify) => {
fastify.decorateRequest('claimGmxFundingFees', claimGmxFundingFees)
fastify.decorateRequest('claimGmxPriceImpact', claimGmxPriceImpact)
fastify.decorateRequest('getGmxPriceImpactRebates', getGmxPriceImpactRebates)
fastify.decorateRequest('claimGmxUiFees', claimGmxUiFees)
// Pre-populate and refresh the markets cache on startup
fastify.addHook('onReady', async () => {
@@ -1548,4 +1555,99 @@ export async function getGmxPriceImpactRebates(
return handleError(this, reply, error, 'gmx/get-price-impact-rebates');
}
}
/**
* Implementation function to claim UI fees
* @param sdk The GMX SDK client
* @returns Transaction hash
*/
export const claimGmxUiFeesImpl = async (
sdk: GmxSdk
): Promise<string> => {
try {
// Get all markets and tokens data
const { marketsInfoData } = await getMarketsInfoWithCache(sdk);
if (!marketsInfoData) {
throw new Error("No markets info data available");
}
const marketAddresses = Object.keys(marketsInfoData);
if (marketAddresses.length === 0) {
throw new Error("No markets available");
}
const allMarketAddresses: string[] = [];
const allTokenAddresses: string[] = [];
// Build arrays of ALL markets and tokens for UI fee claiming
Object.entries(marketsInfoData).forEach(([marketAddress, marketInfo]) => {
// Add market address for long token
allMarketAddresses.push(marketAddress);
allTokenAddresses.push(marketInfo.longToken.address);
// Add market address for short token (if different from long token)
if (marketInfo.longToken.address !== marketInfo.shortToken.address) {
allMarketAddresses.push(marketAddress);
allTokenAddresses.push(marketInfo.shortToken.address);
}
});
if (allMarketAddresses.length === 0) {
throw new Error("No market/token pairs found for UI fee claiming");
}
// Get the ExchangeRouter contract address
const exchangeRouterAddress = getContract(sdk.chainId, "ExchangeRouter");
console.log("UI fees - marketAddresses", allMarketAddresses);
console.log("UI fees - tokenAddresses", allTokenAddresses);
console.log("UI fees - receiver account", sdk.account);
// Execute the claim UI fees transaction using sdk.callContract
await sdk.callContract(
exchangeRouterAddress,
abis.ExchangeRouter as Abi,
"claimUiFees",
[allMarketAddresses, allTokenAddresses, sdk.account]
);
return "ui_fees_claimed"; // Return a success indicator
} catch (error) {
console.error('Error claiming UI fees:', error);
throw new Error(`Failed to claim UI fees: ${error instanceof Error ? error.message : 'Unknown error'}`);
}
};
/**
* Claims UI fees on GMX
* @param this The FastifyRequest instance
* @param reply The FastifyReply instance
* @param account The wallet address of the user
* @returns The response object with success status and transaction hash
*/
export async function claimGmxUiFees(
this: FastifyRequest,
reply: FastifyReply,
account: string
) {
try {
// Validate the request parameters
claimUiFeesSchema.parse({ account });
// Get client for the address
const sdk = await this.getClientForAddress(account);
// Call the implementation function
const hash = await claimGmxUiFeesImpl(sdk);
return {
success: true,
hash
};
} catch (error) {
return handleError(this, reply, error, 'gmx/claim-ui-fees');
}
}

View File

@@ -0,0 +1,92 @@
import {test} from 'node:test';
import assert from 'node:assert';
import {claimGmxUiFeesImpl, getClientForAddress} from '../../src/plugins/custom/gmx.js';
test('GMX Claim UI Fees Implementation', async (t) => {
const testAccount = '0xbBA4eaA534cbD0EcAed5E2fD6036Aec2E7eE309f';
t.test('should claim UI fees for valid account', async () => {
try {
// Get GMX SDK client for the test account
const sdk = await getClientForAddress(testAccount);
// Call the implementation function directly
const result = await claimGmxUiFeesImpl(sdk);
// Validate response
assert.ok(typeof result === 'string', 'Result should be a string');
assert.strictEqual(result, 'ui_fees_claimed', 'Should return success indicator');
console.log('UI fees claim result:', result);
} catch (error) {
console.warn('Expected error in test environment:', error.message);
assert.ok(error instanceof Error, 'Should throw an Error instance');
// In test environment, this will likely fail due to no claimable fees or network issues
// This is expected behavior in a test environment
}
});
t.test('should handle SDK initialization for different account', async () => {
try {
const differentAccount = '0x1234567890123456789012345678901234567890';
// Get GMX SDK client for a different account
const sdk = await getClientForAddress(differentAccount);
// Verify SDK was created with correct account
assert.strictEqual(sdk.account.toLowerCase(), differentAccount.toLowerCase(), 'SDK should be initialized with correct account');
// Call the implementation function
const result = await claimGmxUiFeesImpl(sdk);
assert.ok(typeof result === 'string', 'Result should be a string');
console.log('Different account claim result:', result);
} catch (error) {
console.warn('Expected error in test environment:', error.message);
assert.ok(error instanceof Error, 'Should throw an Error instance');
}
});
t.test('should handle errors gracefully', async () => {
try {
// Test with invalid/empty account to see error handling
const invalidAccount = '0x0000000000000000000000000000000000000000';
const sdk = await getClientForAddress(invalidAccount);
await claimGmxUiFeesImpl(sdk);
// If we reach here without error, that's also valid
console.log('Invalid account test passed without error');
} catch (error) {
// Expected to fail with invalid account
console.log('Error handling test - caught expected error:', error.message);
assert.ok(error instanceof Error, 'Should throw an Error instance');
assert.ok(error.message.includes('Failed to claim UI fees') ||
error.message.includes('No markets') ||
error.message.includes('network') ||
error.message.includes('RPC'), 'Should have meaningful error message');
}
});
t.test('should validate SDK client creation', async () => {
try {
const sdk = await getClientForAddress(testAccount);
// Validate SDK properties
assert.ok(sdk, 'SDK should be created');
assert.ok(sdk.account, 'SDK should have account property');
assert.ok(sdk.chainId, 'SDK should have chainId property');
assert.strictEqual(sdk.account.toLowerCase(), testAccount.toLowerCase(), 'SDK account should match input');
console.log('SDK validation passed for account:', sdk.account);
console.log('SDK chainId:', sdk.chainId);
} catch (error) {
console.warn('SDK creation error:', error.message);
assert.ok(error instanceof Error, 'Should throw an Error instance');
}
});
});