* Filter everything with users * Fix backtests and user management * Add cursor rules * Fix backtest and bots * Update configs names * Sign until unauth * Setup delegate * Setup delegate and sign * refact * Enhance Privy signature generation with improved cryptographic methods * Add Fastify backend * Add Fastify backend routes for privy * fix privy signing * fix privy client * Fix tests * add gmx core * fix merging sdk * Fix tests * add gmx core * add gmx core * add privy to boilerplate * clean * fix * add fastify * Remove Managing.Fastify submodule * Add Managing.Fastify as regular directory instead of submodule * Update .gitignore to exclude Managing.Fastify dist and node_modules directories * Add token approval functionality to Privy plugin - Introduced a new endpoint `/approve-token` for approving ERC20 tokens. - Added `approveToken` method to the Privy plugin for handling token approvals. - Updated `signPrivyMessage` to differentiate between message signing and token approval requests. - Enhanced the plugin with additional schemas for input validation. - Included new utility functions for token data retrieval and message construction. - Updated tests to verify the new functionality and ensure proper request decoration. * Add PrivyApproveTokenResponse model for token approval response - Created a new class `PrivyApproveTokenResponse` to encapsulate the response structure for token approval requests. - The class includes properties for `Success` status and a transaction `Hash`. * Refactor trading commands and enhance API routes - Updated `OpenPositionCommandHandler` to use asynchronous methods for opening trades and canceling orders. - Introduced new Fastify routes for opening positions and canceling orders with appropriate request validation. - Modified `EvmManager` to handle both Privy and non-Privy wallet operations, utilizing the Fastify API for Privy wallets. - Adjusted test configurations to reflect changes in account types and added helper methods for testing Web3 proxy services. * Enhance GMX trading functionality and update dependencies - Updated `dev:start` script in `package.json` to include the `-d` flag for Fastify. - Upgraded `fastify-cli` dependency to version 7.3.0. - Added `sourceMap` option to `tsconfig.json`. - Refactored GMX plugin to improve position opening logic, including enhanced error handling and validation. - Introduced a new method `getMarketInfoFromTicker` for better market data retrieval. - Updated account type in `PrivateKeys.cs` to use `Privy`. - Adjusted `EvmManager` to utilize the `direction` enum directly for trade direction handling. * Refactor GMX plugin for improved trading logic and market data retrieval - Enhanced the `openGmxPositionImpl` function to utilize the `TradeDirection` enum for trade direction handling. - Introduced `getTokenDataFromTicker` and `getMarketByIndexToken` functions for better market and token data retrieval. - Updated collateral calculation and logging for clarity. - Adjusted `EvmManager` to ensure proper handling of price values in trade requests. * Refactor GMX plugin and enhance testing for position opening - Updated `test:single` script in `package.json` to include TypeScript compilation before running tests. - Removed `this` context from `getClientForAddress` function and replaced logging with `console.error`. - Improved collateral calculation in `openGmxPositionImpl` for better precision. - Adjusted type casting for `direction` in the API route to utilize `TradeDirection` enum. - Added a new test for opening a long position in GMX, ensuring functionality and correctness. * Update sdk * Update * update fastify * Refactor start script in package.json to simplify command execution - Removed the build step from the start script, allowing for a more direct launch of the Fastify server. * Update package.json for Web3Proxy - Changed the name from "Web3Proxy" to "web3-proxy". - Updated version from "0.0.0" to "1.0.0". - Modified the description to "The official Managing Web3 Proxy". * Update Dockerfile for Web3Proxy - Upgraded Node.js base image from 18-alpine to 22.14.0-alpine. - Added NODE_ENV environment variable set to production. * Refactor Dockerfile and package.json for Web3Proxy - Removed the build step from the Dockerfile to streamline the image creation process. - Updated the start script in package.json to include the build step, ensuring the application is built before starting the server. * Add fastify-tsconfig as a development dependency in Dockerfile-web3proxy * Remove fastify-tsconfig extension from tsconfig.json for Web3Proxy * Add PrivyInitAddressResponse model for handling initialization responses - Introduced a new class `PrivyInitAddressResponse` to encapsulate the response structure for Privy initialization, including properties for success status, USDC hash, order vault hash, and error message. * Update * Update * Remove fastify-tsconfig installation from Dockerfile-web3proxy * Add build step to Dockerfile-web3proxy - Included `npm run build` in the Dockerfile to ensure the application is built during the image creation process. * Update * approvals * Open position from front embedded wallet * Open position from front embedded wallet * Open position from front embedded wallet * Fix call contracts * Fix limit price * Close position * Fix close position * Fix close position * add pinky * Refactor position handling logic * Update Dockerfile-pinky to copy package.json and source code from the correct directory * Implement password protection modal and enhance UI with new styles; remove unused audio elements and update package dependencies. * add cancel orders * Update callContract function to explicitly cast account address as Address type * Update callContract function to cast transaction parameters as any type for compatibility * Cast transaction parameters as any type in approveTokenImpl for compatibility * Cast wallet address and transaction parameters as Address type in approveTokenImpl for type safety * Add .env configuration file for production setup including database and server settings * Refactor home route to update welcome message and remove unused SDK configuration code * add referral code * fix referral * Add sltp * Fix typo * Fix typo * setup sltp on backtend * get orders * get positions with slp * fixes * fixes close position * fixes * Remove MongoDB project references from Dockerfiles for managing and worker APIs * Comment out BotManagerWorker service registration and remove MongoDB project reference from Dockerfile * fixes
294 lines
11 KiB
C#
294 lines
11 KiB
C#
using Managing.Application.Abstractions.Repositories;
|
|
using Managing.Application.Abstractions.Services;
|
|
using Managing.Domain.Accounts;
|
|
using Managing.Domain.Candles;
|
|
using Managing.Domain.Statistics;
|
|
using Managing.Domain.Trades;
|
|
using Managing.Infrastructure.Exchanges.Abstractions;
|
|
using Microsoft.Extensions.Logging;
|
|
using static Managing.Common.Enums;
|
|
|
|
namespace Managing.Infrastructure.Exchanges
|
|
{
|
|
public class ExchangeService : IExchangeService
|
|
{
|
|
private readonly ILogger<ExchangeService> _logger;
|
|
private readonly ICandleRepository _candleRepository;
|
|
private readonly IEnumerable<IExchangeProcessor> _exchangeProcessor;
|
|
|
|
public ExchangeService(ILogger<ExchangeService> logger, ICandleRepository candleRepository,
|
|
IEnumerable<IExchangeProcessor> processor)
|
|
{
|
|
_logger = logger;
|
|
_candleRepository = candleRepository;
|
|
_exchangeProcessor = processor;
|
|
}
|
|
|
|
#region Trades
|
|
|
|
public async Task<Trade> OpenTrade(
|
|
Account account,
|
|
Ticker ticker,
|
|
TradeDirection direction,
|
|
decimal price,
|
|
decimal quantity,
|
|
decimal? leverage = null,
|
|
TradeType tradeType = TradeType.Limit,
|
|
bool reduceOnly = false,
|
|
bool isForPaperTrading = false,
|
|
DateTime? currentDate = null,
|
|
bool ioc = true,
|
|
decimal? stopLossPrice = null,
|
|
decimal? takeProfitPrice = null)
|
|
{
|
|
_logger.LogInformation(
|
|
$"OpenMarketTrade - {ticker} - Type: {tradeType} - {direction} - Price: {price} - Quantity: {quantity} - Leverage: {leverage} - SL: {stopLossPrice} - TP: {takeProfitPrice}");
|
|
|
|
if (isForPaperTrading)
|
|
{
|
|
return BuildEmptyTrade(ticker, price, quantity, direction, leverage, tradeType, currentDate.Value,
|
|
reduceOnly ? TradeStatus.PendingOpen : TradeStatus.Filled);
|
|
}
|
|
|
|
var processor = GetProcessor(account);
|
|
return await processor.OpenTrade(account, ticker, direction, price, quantity, leverage, tradeType,
|
|
reduceOnly, isForPaperTrading, currentDate, ioc, stopLossPrice, takeProfitPrice);
|
|
}
|
|
|
|
private IExchangeProcessor GetProcessor(Account account)
|
|
{
|
|
var exchange = EvmTradingExchangesList.Contains(account.Exchange) ? TradingExchanges.Evm : account.Exchange;
|
|
return _exchangeProcessor.First(e => e.Exchange() == exchange);
|
|
}
|
|
|
|
private static List<TradingExchanges> EvmTradingExchangesList
|
|
{
|
|
get
|
|
{
|
|
var eligibileProcessors = new List<TradingExchanges>()
|
|
{
|
|
TradingExchanges.Evm, TradingExchanges.GmxV2
|
|
};
|
|
return eligibileProcessors;
|
|
}
|
|
}
|
|
|
|
|
|
private bool IsEvmExchange(Account account)
|
|
{
|
|
return EvmTradingExchangesList.Contains(account.Exchange);
|
|
}
|
|
|
|
public Trade BuildEmptyTrade(Ticker ticker, decimal price, decimal quantity, TradeDirection direction,
|
|
decimal? leverage, TradeType tradeType,
|
|
DateTime dateTime, TradeStatus tradeStatus)
|
|
{
|
|
return new Trade(dateTime, direction, tradeStatus, tradeType, ticker, quantity, price, leverage,
|
|
Guid.NewGuid().ToString(), "EmptyTrade");
|
|
}
|
|
|
|
public async Task<Trade> OpenStopLoss(Account account, Ticker ticker, TradeDirection originalDirection,
|
|
decimal stopLossPrice,
|
|
decimal quantity, bool isForPaperTrading = false, DateTime? currentDate = null)
|
|
{
|
|
return await OpenTrade(
|
|
account,
|
|
ticker,
|
|
originalDirection == TradeDirection.Long ? TradeDirection.Short : TradeDirection.Long,
|
|
stopLossPrice,
|
|
quantity,
|
|
tradeType: TradeType.StopLoss,
|
|
isForPaperTrading: isForPaperTrading,
|
|
currentDate: currentDate,
|
|
reduceOnly: true);
|
|
}
|
|
|
|
public async Task<Trade> OpenTakeProfit(Account account, Ticker ticker, TradeDirection originalDirection,
|
|
decimal takeProfitPrice,
|
|
decimal quantity, bool isForPaperTrading = false, DateTime? currentDate = null)
|
|
{
|
|
return await OpenTrade(
|
|
account,
|
|
ticker,
|
|
originalDirection == TradeDirection.Long ? TradeDirection.Short : TradeDirection.Long,
|
|
takeProfitPrice,
|
|
quantity,
|
|
tradeType: TradeType.TakeProfit,
|
|
isForPaperTrading: isForPaperTrading,
|
|
currentDate: currentDate,
|
|
reduceOnly: true);
|
|
}
|
|
|
|
public async Task<Trade> ClosePosition(Account account, Position position, decimal lastPrice,
|
|
bool isForPaperTrading = false)
|
|
{
|
|
var direction = position.OriginDirection == TradeDirection.Long
|
|
? TradeDirection.Short
|
|
: TradeDirection.Long;
|
|
|
|
if (isForPaperTrading)
|
|
{
|
|
var fake = BuildEmptyTrade(position.Open.Ticker,
|
|
lastPrice,
|
|
position.Open.Quantity,
|
|
direction,
|
|
position.Open.Leverage,
|
|
TradeType.Market,
|
|
position.Open.Date,
|
|
TradeStatus.Filled);
|
|
return fake;
|
|
}
|
|
|
|
var processor = GetProcessor(account);
|
|
var closedTrade = await processor.OpenTrade(
|
|
account,
|
|
position.Ticker,
|
|
direction,
|
|
lastPrice,
|
|
position.Open.Quantity,
|
|
position.Open.Leverage,
|
|
tradeType: TradeType.Market,
|
|
reduceOnly: true);
|
|
|
|
if (!IsEvmExchange(account))
|
|
{
|
|
// Only use for non-EVM exchanges since the call to the blockchain is async
|
|
closedTrade.Price = processor.GetTrade(account, closedTrade.ExchangeOrderId, closedTrade.Ticker).Result
|
|
.Price;
|
|
}
|
|
|
|
return closedTrade;
|
|
}
|
|
|
|
#endregion
|
|
|
|
|
|
public async Task<bool> CancelOrder(Account account, Ticker ticker)
|
|
{
|
|
var processor = GetProcessor(account);
|
|
return await processor.CancelOrder(account, ticker);
|
|
}
|
|
|
|
public async Task<Trade> GetTrade(Account account, string orderId, Ticker ticker)
|
|
{
|
|
var processor = GetProcessor(account);
|
|
return await processor.GetTrade(account, orderId, ticker);
|
|
}
|
|
|
|
public async Task<Trade> GetTrade(string reference, string orderId, Ticker ticker)
|
|
{
|
|
var processor = _exchangeProcessor.First(e => e.Exchange() == TradingExchanges.Evm);
|
|
return await processor.GetTrade(reference, orderId, ticker);
|
|
}
|
|
|
|
public Task<List<FundingRate>> GetFundingRates()
|
|
{
|
|
var processor = _exchangeProcessor.First(e => e.Exchange() == TradingExchanges.Evm);
|
|
return processor.GetFundingRates();
|
|
}
|
|
|
|
public async Task<List<Trade>> GetTrades(Account account, Ticker ticker)
|
|
{
|
|
var processor = GetProcessor(account);
|
|
return await processor.GetTrades(account, ticker);
|
|
}
|
|
|
|
public async Task<List<Candle>> GetCandles(Account account, Ticker ticker, DateTime startDate,
|
|
Timeframe timeframe)
|
|
{
|
|
var processor = GetProcessor(account);
|
|
return await processor.GetCandles(account, ticker, startDate, timeframe);
|
|
}
|
|
|
|
public async Task<List<Candle>> GetCandlesInflux(TradingExchanges exchange, Ticker ticker, DateTime startDate,
|
|
Timeframe timeframe)
|
|
{
|
|
var candlesFromRepo = await _candleRepository.GetCandles(exchange, ticker, timeframe, startDate);
|
|
return candlesFromRepo.ToList();
|
|
}
|
|
|
|
public async Task<List<Candle>> GetCandlesInflux(TradingExchanges exchange, Ticker ticker, DateTime startDate,
|
|
Timeframe timeframe, DateTime endDate)
|
|
{
|
|
var candlesFromRepo = await _candleRepository.GetCandles(exchange, ticker, timeframe, startDate, endDate);
|
|
return candlesFromRepo.ToList();
|
|
}
|
|
|
|
public async Task<decimal> GetBalance(Account account, bool isForPaperTrading = false)
|
|
{
|
|
if (isForPaperTrading)
|
|
{
|
|
return 1000;
|
|
}
|
|
|
|
var processor = GetProcessor(account);
|
|
return await processor.GetBalance(account);
|
|
}
|
|
|
|
public decimal GetFee(Account account, bool isForPaperTrading = false)
|
|
{
|
|
var processor = GetProcessor(account);
|
|
return processor.GetFee(account);
|
|
}
|
|
|
|
public decimal GetPrice(Account account, Ticker ticker, DateTime date)
|
|
{
|
|
var processor = GetProcessor(account);
|
|
return processor.GetPrice(account, ticker, date);
|
|
}
|
|
|
|
public Candle GetCandle(Account account, Ticker ticker, DateTime date)
|
|
{
|
|
var processor = GetProcessor(account);
|
|
return processor.GetCandle(account, ticker, date);
|
|
}
|
|
|
|
public async Task<decimal> GetQuantityInPosition(Account account, Ticker ticker)
|
|
{
|
|
var processor = GetProcessor(account);
|
|
return await processor.GetQuantityInPosition(account, ticker);
|
|
}
|
|
|
|
public decimal GetVolume(Account account, Ticker ticker)
|
|
{
|
|
var processor = GetProcessor(account);
|
|
return processor.GetVolume(account, ticker);
|
|
}
|
|
|
|
public async Task<List<Ticker>> GetTickers(Timeframe timeframe)
|
|
{
|
|
var tickers =
|
|
await _candleRepository.GetTickersAsync(TradingExchanges.Evm, timeframe, DateTime.UtcNow.AddDays(-2));
|
|
return tickers.ToList();
|
|
}
|
|
|
|
public decimal GetBestPrice(Account account, Ticker ticker, decimal lastPrice, decimal quantity,
|
|
TradeDirection direction)
|
|
{
|
|
if (IsEvmExchange(account))
|
|
{
|
|
return GetPrice(account, ticker, DateTime.UtcNow);
|
|
}
|
|
|
|
return GetOrderbook(account, ticker).GetBestPrice(direction, quantity);
|
|
}
|
|
|
|
public Orderbook GetOrderbook(Account account, Ticker ticker)
|
|
{
|
|
var processor = GetProcessor(account);
|
|
return processor.GetOrderbook(account, ticker);
|
|
}
|
|
|
|
public async Task<List<Balance>> GetBalances(Account account, bool isForPaperTrading = false)
|
|
{
|
|
var processor = GetProcessor(account);
|
|
return await processor.GetBalances(account, isForPaperTrading);
|
|
}
|
|
|
|
public async Task<List<Trade>> GetOpenOrders(Account account, Ticker ticker)
|
|
{
|
|
var processor = GetProcessor(account);
|
|
return await processor.GetOrders(account, ticker);
|
|
}
|
|
}
|
|
} |