Add stats and trades to bots

This commit is contained in:
2025-04-25 16:59:08 +07:00
parent 9b0505991f
commit d5dead3d8f
3 changed files with 84 additions and 17 deletions

View File

@@ -727,7 +727,8 @@ public class TradingBot : Bot, ITradingBot
{
await LogInformation($"Position {signalIdentifier} is now {positionStatus}");
Positions.First(p => p.SignalIdentifier == signalIdentifier).Status = positionStatus;
SetSignalStatus(signalIdentifier, SignalStatus.Expired);
SetSignalStatus(signalIdentifier,
positionStatus == PositionStatus.Filled ? SignalStatus.PositionOpen : SignalStatus.Expired);
}
private void SetSignalStatus(string signalIdentifier, SignalStatus signalStatus)

View File

@@ -1,15 +1,11 @@
import fp from 'fastify-plugin'
import {FastifyReply, FastifyRequest, FastifyInstance} from 'fastify'
import {FastifyInstance, FastifyReply, FastifyRequest} from 'fastify'
import {z} from 'zod'
import canonicalize from 'canonicalize'
import crypto from 'crypto'
import {PrivyClient} from '@privy-io/server-auth'
import {ethers} from 'ethers'
import dotenv from 'dotenv'
// Load environment variables
dotenv.config()
import Token from '../../generated/gmxsdk/abis/Token.json' with {type: 'json'}
import {ARBITRUM} from '../../generated/gmxsdk/configs/chains.js'
import {TOKENS} from '../../generated/gmxsdk/configs/tokens.js'
@@ -18,6 +14,9 @@ import { getClientForAddress, getTokenDataFromTicker } from './gmx.js'
import {Ticker} from '../../generated/ManagingApiTypes.js'
import {Address} from 'viem'
// Load environment variables
dotenv.config()
/**
* Privy Plugin
*
@@ -72,7 +71,7 @@ export const getPrivyClient = (fastify?: FastifyInstance): PrivyClient => {
walletApi: {
authorizationPrivateKey: authKey
}
}
},
);
};

View File

@@ -1,10 +1,14 @@
import {ArrowDownIcon, ArrowUpIcon} from '@heroicons/react/solid'
import React, {useEffect, useState} from 'react'
import type { TradingBot } from '../../../generated/ManagingApi'
import { PositionStatus, TradeDirection } from '../../../generated/ManagingApi'
import useApiUrlStore from '../../../app/store/apiStore'
import type {PlatformSummaryViewModel, TradingBot} from '../../../generated/ManagingApi'
import {DataClient, PositionStatus, TradeDirection} from '../../../generated/ManagingApi'
import type {IAccountBalanceProps} from '../../../global/type'
// Time filter options matching backend
const TIME_FILTERS = ['24H', '3D', '1W', '1M', '1Y', 'Total']
function GetGlobalWinrate(bots: TradingBot[]) {
if (bots == null || bots == undefined || bots.length == 0) {
return 0
@@ -55,6 +59,9 @@ function GetPositionCount(
const Summary: React.FC<IAccountBalanceProps> = ({ bots }) => {
const [globalPnl, setGlobalPnl] = useState<number>(0)
const [globalWinrate, setGlobalWinrate] = useState<number>(0)
const [selectedTimeFilter, setSelectedTimeFilter] = useState<string>('Total')
const [platformStats, setPlatformStats] = useState<PlatformSummaryViewModel | null>(null)
const [isLoading, setIsLoading] = useState<boolean>(false)
const [openLong, setLong] = useState<number>(0)
const [openShort, setShort] = useState<number>(0)
@@ -62,6 +69,8 @@ const Summary: React.FC<IAccountBalanceProps> = ({ bots }) => {
const [closedLong, setClosedLong] = useState<number>(0)
const [closedShort, setClosedShort] = useState<number>(0)
const { apiUrl } = useApiUrlStore()
useEffect(() => {
if (bots) {
const pnl = bots.reduce((acc, bot) => {
@@ -86,9 +95,67 @@ const Summary: React.FC<IAccountBalanceProps> = ({ bots }) => {
}
}, [bots])
// Fetch platform summary data
useEffect(() => {
const fetchPlatformStats = async () => {
setIsLoading(true)
try {
const dataClient = new DataClient({}, apiUrl)
const data = await dataClient.data_GetPlatformSummary(selectedTimeFilter)
setPlatformStats(data)
} catch (error) {
console.error('Error fetching platform stats:', error)
} finally {
setIsLoading(false)
}
}
fetchPlatformStats()
}, [apiUrl, selectedTimeFilter])
return (
<div className="p-4">
<div className="stats bg-primary text-primary-content mb-4"></div>
<div className="mb-4 flex justify-between items-center">
<h2 className="text-xl font-bold">Platform Overview</h2>
<div className="join">
{TIME_FILTERS.map((filter) => (
<button
key={filter}
className={`btn btn-sm join-item ${selectedTimeFilter === filter ? 'btn-primary' : 'btn-ghost'}`}
onClick={() => setSelectedTimeFilter(filter)}
>
{filter}
</button>
))}
</div>
</div>
<div className="stats bg-primary text-primary-content mb-4">
<div className="stat">
<div className="stat-title">Total Agents</div>
<div className="stat-value">{platformStats?.totalAgents ?? 0}</div>
</div>
<div className="stat">
<div className="stat-title">Active Strategies</div>
<div className="stat-value">{platformStats?.totalActiveStrategies ?? 0}</div>
</div>
<div className="stat">
<div className="stat-title">Total Platform PnL</div>
<div className="stat-value">{(platformStats?.totalPlatformPnL ?? 0).toFixed(2)} $</div>
</div>
<div className="stat">
<div className="stat-title">Volume (Total)</div>
<div className="stat-value">{(platformStats?.totalPlatformVolume ?? 0).toFixed(2)} $</div>
</div>
<div className="stat">
<div className="stat-title">Volume (24h)</div>
<div className="stat-value">{(platformStats?.totalPlatformVolumeLast24h ?? 0).toFixed(2)} $</div>
</div>
</div>
<div className="stats bg-primary text-primary-content">
<div className="stat">