OctoBot-Trading 2.4.74__tar.gz → 2.4.76__tar.gz
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/CHANGELOG.md +12 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76/OctoBot_Trading.egg-info}/PKG-INFO +2 -2
- {OctoBot-Trading-2.4.74/OctoBot_Trading.egg-info → OctoBot-Trading-2.4.76}/PKG-INFO +2 -2
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/README.md +1 -1
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/__init__.py +1 -1
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/constants.py +5 -2
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/errors.py +6 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/__init__.py +2 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/abstract_exchange.py +26 -1
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/connectors/ccxt/ccxt_connector.py +6 -2
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/traders/trader.py +3 -1
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/types/rest_exchange.py +21 -7
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/util/__init__.py +2 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/util/exchange_util.py +3 -3
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/modes_util.py +21 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/channel/orders.py +19 -1
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/channel/orders_updater.py +1 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/channel/orders_updater_simulator.py +4 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/order_factory.py +3 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/orders_storage_operations.py +28 -4
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/storage/orders_storage.py +8 -2
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/util/test_tools/exchanges_test_tools.py +37 -2
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/util/test_exchange_util.py +8 -8
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/modes/test_modes_util.py +58 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/LICENSE +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/MANIFEST.in +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/OctoBot_Trading.egg-info/SOURCES.txt +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/OctoBot_Trading.egg-info/dependency_links.txt +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/OctoBot_Trading.egg-info/not-zip-safe +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/OctoBot_Trading.egg-info/requires.txt +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/OctoBot_Trading.egg-info/top_level.txt +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/api/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/api/channels.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/api/contracts.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/api/exchange.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/api/modes.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/api/orders.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/api/portfolio.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/api/positions.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/api/profitability.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/api/storage.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/api/symbol_data.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/api/trader.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/api/trades.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/enums.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_channel.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/contracts/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/contracts/contract_factory.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/contracts/future_contract.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/contracts/margin_contract.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/exchange_symbol_data.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/exchange_symbols_data.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/funding/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/funding/channel/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/funding/channel/funding.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/funding/channel/funding_updater.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/funding/channel/funding_updater_simulator.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/funding/funding_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/kline/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/kline/channel/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/kline/channel/kline.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/kline/channel/kline_updater.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/kline/channel/kline_updater_simulator.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/kline/kline_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/ohlcv/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/ohlcv/candles_adapter.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/ohlcv/candles_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/ohlcv/channel/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/ohlcv/channel/ohlcv.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/ohlcv/channel/ohlcv_updater.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/ohlcv/channel/ohlcv_updater_simulator.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/ohlcv/preloaded_candles_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/order_book/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/order_book/channel/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/order_book/channel/order_book.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/order_book/channel/order_book_updater.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/order_book/channel/order_book_updater_simulator.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/order_book/order_book_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/prices/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/prices/channel/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/prices/channel/price.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/prices/channel/prices_updater.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/prices/channel/prices_updater_simulator.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/prices/price_events_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/prices/prices_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/recent_trades/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/recent_trades/channel/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/recent_trades/channel/recent_trade.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/recent_trades/channel/recent_trade_updater.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/recent_trades/channel/recent_trade_updater_simulator.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/recent_trades/recent_trades_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/ticker/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/ticker/channel/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/ticker/channel/ticker.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/ticker/channel/ticker_updater.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/ticker/channel/ticker_updater_simulator.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchange_data/ticker/ticker_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/abstract_websocket_exchange.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/adapters/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/adapters/abstract_adapter.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/config/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/config/backtesting_exchange_config.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/config/exchange_config_data.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/connectors/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/connectors/ccxt/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/connectors/ccxt/ccxt_adapter.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/connectors/ccxt/ccxt_client_util.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/connectors/ccxt/ccxt_clients_cache.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/connectors/ccxt/ccxt_websocket_connector.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/connectors/ccxt/constants.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/connectors/ccxt/enums.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/connectors/simulator/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/connectors/simulator/ccxt_client_simulation.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/connectors/simulator/exchange_simulator_adapter.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/connectors/simulator/exchange_simulator_connector.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/exchange_builder.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/exchange_channels.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/exchange_details.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/exchange_factory.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/exchange_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/exchange_websocket_factory.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/exchanges.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/implementations/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/implementations/default_rest_exchange.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/implementations/default_websocket_exchange.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/implementations/exchange_simulator.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/traders/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/traders/trader_simulator.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/types/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/types/websocket_exchange.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/util/exchange_market_status_fixer.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/util/symbol_details.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/util/websockets_util.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/abstract_trading_mode.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/channel/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/channel/abstract_mode_consumer.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/channel/abstract_mode_producer.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/channel/mode.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/mode_activity.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/mode_config.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/modes_factory.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/script_keywords/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/script_keywords/basic_keywords/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/script_keywords/basic_keywords/account_balance.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/script_keywords/basic_keywords/amount.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/script_keywords/basic_keywords/configuration.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/script_keywords/basic_keywords/position.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/script_keywords/basic_keywords/price.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/script_keywords/basic_keywords/run_persistence.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/script_keywords/basic_keywords/trading_signals.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/script_keywords/basic_keywords/user_inputs.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/script_keywords/context_management.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/script_keywords/dsl/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/script_keywords/dsl/quantity.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/script_keywords/dsl/values.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/scripted_trading_mode/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/modes/scripted_trading_mode/abstract_scripted_trading_mode.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/octobot_channel_consumer.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/exchange_personal_data.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/channel/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/decimal_order_adapter.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/groups/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/groups/balanced_take_profit_and_stop_order_group.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/groups/group_util.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/groups/one_cancels_the_other_order_group.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/order_adapter.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/order_group.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/order_state.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/order_util.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/orders_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/states/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/states/cancel_order_state.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/states/close_order_state.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/states/fill_order_state.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/states/open_order_state.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/states/order_state_factory.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/states/pending_creation_chained_order_state.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/states/pending_creation_order_state.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/types/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/types/limit/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/types/limit/buy_limit_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/types/limit/limit_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/types/limit/sell_limit_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/types/limit/stop_loss_limit_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/types/limit/stop_loss_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/types/limit/take_profit_limit_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/types/limit/take_profit_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/types/market/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/types/market/buy_market_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/types/market/market_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/types/market/sell_market_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/types/trailing/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/types/trailing/trailing_stop_limit_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/types/trailing/trailing_stop_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/types/unknown_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/orders/types/unsupported_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/asset.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/assets/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/assets/future_asset.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/assets/margin_asset.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/assets/spot_asset.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/channel/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/channel/balance.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/channel/balance_updater.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/channel/balance_updater_simulator.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/history/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/history/historical_asset_value.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/history/historical_asset_value_factory.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/history/historical_portfolio_value_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/portfolio.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/portfolio_factory.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/portfolio_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/portfolio_profitability.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/portfolio_util.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/portfolio_value_holder.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/sub_portfolio.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/types/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/types/future_portfolio.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/types/margin_portfolio.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/types/spot_portfolio.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/portfolios/value_converter.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/positions/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/positions/channel/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/positions/channel/positions.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/positions/channel/positions_updater.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/positions/channel/positions_updater_simulator.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/positions/position.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/positions/position_factory.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/positions/position_state.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/positions/position_util.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/positions/positions_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/positions/states/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/positions/states/active_position_state.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/positions/states/idle_position_state.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/positions/states/liquidate_position_state.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/positions/states/position_state_factory.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/positions/types/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/positions/types/inverse_position.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/positions/types/linear_position.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/state.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/trades/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/trades/channel/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/trades/channel/trades.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/trades/channel/trades_updater.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/trades/trade.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/trades/trade_factory.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/trades/trade_pnl.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/trades/trades_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/trades/trades_util.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/transactions/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/transactions/transaction.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/transactions/transaction_factory.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/transactions/transactions_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/transactions/types/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/transactions/types/blockchain_transaction.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/transactions/types/fee_transaction.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/transactions/types/realised_pnl_transaction.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/personal_data/transactions/types/transfer_transaction.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/signals/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/signals/channel/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/signals/channel/remote_trading_signal.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/signals/channel/remote_trading_signal_channel_factory.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/signals/channel/signal_producer.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/signals/signal_creation.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/signals/trading_signal_bundle_builder.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/signals/util.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/storage/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/storage/abstract_storage.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/storage/candles_storage.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/storage/portfolio_storage.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/storage/storage_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/storage/trades_storage.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/storage/transactions_storage.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/storage/util.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/supervisors/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/supervisors/abstract_portfolio_supervisor.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/supervisors/abstract_supervisor.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/util/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/util/config_util.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/util/initializable.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/util/initialization_util.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/util/simulator_updater_utils.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/util/test_tools/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/util/test_tools/exchange_data.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/util/test_tools/spot_rest_exchange_test_tools.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/util/test_tools/websocket_test_tools.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/requirements.txt +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/setup.cfg +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/setup.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/api/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/api/test_channels.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/api/test_exchange.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/api/test_modes.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/api/test_orders.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/api/test_portfolio.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/api/test_profitability.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/api/test_symbol_data.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/api/test_trader.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/api/test_trades.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/cli/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/contracts/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/contracts/test_future_contract.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/contracts/test_margin_contract.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/funding/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/funding/test_funding_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/kline/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/kline/test_kline_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/ohlcv/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/ohlcv/test_candles_adapter.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/ohlcv/test_candles_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/order_book/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/order_book/test_order_book_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/prices/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/prices/test_price_events_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/prices/test_prices_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/recent_trades/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/recent_trades/test_recent_trades_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/test_exchange_symbols_data.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/ticker/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchange_data/ticker/test_ticker_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/connectors/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/connectors/ccxt/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/connectors/ccxt/mock_exchanges_data.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/connectors/ccxt/test_ccxt_connector.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/implementations/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/implementations/test_default_rest_exchange.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/implementations/test_default_websocket_exchange.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/test_abstract_exchange.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/test_abstract_websocket_exchange.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/test_exchange_builder.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/test_exchange_config_data.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/test_exchange_factory.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/test_exchange_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/test_exchange_simulator.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/test_exchanges.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/traders/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/traders/test_trader.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/types/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/types/test_websocket_exchange.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/util/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/util/test_exchange_market_status_fixer.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/modes/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/modes/script_keywords/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/modes/script_keywords/basic_keywords/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/modes/script_keywords/basic_keywords/test_account_balance.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/modes/script_keywords/basic_keywords/test_amount.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/modes/script_keywords/basic_keywords/test_position.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/modes/script_keywords/basic_keywords/test_price.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/modes/script_keywords/dsl/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/modes/script_keywords/dsl/test_quantity.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/modes/test_abstract_mode_consumer.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/modes/test_abstract_trading_mode.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/groups/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/groups/test_balanced_take_profit_and_stop_order_group.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/groups/test_group_util.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/groups/test_one_cancels_the_other_order_group.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/states/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/states/test_cancel_order_state.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/states/test_close_order_state.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/states/test_fill_order_state.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/states/test_open_order_state.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/states/test_order_state.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/states/test_order_state_factory.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/states/test_pending_creation_chained_order_state.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/states/test_pending_creation_order_state.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/test_decimal_order_adapter.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/test_double_filled_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/test_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/test_order_adapter.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/test_order_factory.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/test_order_util.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/test_orders_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/test_orders_storage_operations.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/types/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/types/limit/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/types/limit/test_buy_limit_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/types/limit/test_limit_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/types/limit/test_sell_limit_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/types/limit/test_stop_loss_limit_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/types/limit/test_stop_loss_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/types/limit/test_take_profit_limit_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/types/limit/test_take_profit_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/types/market/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/types/market/test_buy_market_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/types/market/test_sell_market_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/types/test_unknown_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/types/trailing/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/types/trailing/test_trailing_stop_limit_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/orders/types/trailing/test_trailing_stop_order.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/portfolios/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/portfolios/assets/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/portfolios/assets/test_future_asset.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/portfolios/assets/test_margin_asset.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/portfolios/assets/test_spot_asset.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/portfolios/history/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/portfolios/history/test_historical_asset_value_factory.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/portfolios/history/test_historical_portfolio_value_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/portfolios/test_asset.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/portfolios/test_portfolio.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/portfolios/test_portfolio_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/portfolios/test_portfolio_profitability.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/portfolios/test_portfolio_value_holder.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/portfolios/test_value_converter.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/portfolios/types/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/portfolios/types/test_future_portfolio.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/portfolios/types/test_margin_portfolio.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/portfolios/types/test_spot_portfolio.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/positions/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/positions/channel/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/positions/states/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/positions/test_position.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/positions/test_position_factory.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/positions/test_positions_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/positions/types/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/positions/types/test_inverse_position.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/positions/types/test_linear_position.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/trades/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/trades/test_trade_factory.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/trades/test_trade_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/trades/test_trade_pnl.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/trades/test_trade_util.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/transactions/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/transactions/test_transaction_factory.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/personal_data/transactions/test_transactions_manager.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/signals/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/signals/test_trading_signal_bundle_builder.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/signals/test_util.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/test_utils/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/test_utils/order_util.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/test_utils/random_numbers.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/util/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/util/test_config_util.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/__init__.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/real_exchange_tester.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/real_futures_exchange_tester.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_ascendex.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_binance.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_binance_futures.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_bingx.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_bitfinex.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_bitget.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_bithumb.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_bitso.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_bitstamp.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_bybit.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_bybit_futures.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_coinbase.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_coinex.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_cryptocom.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_gateio.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_hitbtc.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_hollaex.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_htx.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_kraken.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_kucoin.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_kucoin_futures.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_mexc.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_ndax.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_okcoin.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_okx.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_okx_futures.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_phemex.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_poloniex.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_upbit.py +0 -0
- {OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests_additional/real_exchanges/test_wavesexchange.py +0 -0
@@ -4,6 +4,18 @@ All notable changes to this project will be documented in this file.
|
|
4
4
|
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
5
5
|
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
6
6
|
|
7
|
+
## [2.4.76] - 2024-04-12
|
8
|
+
### Added
|
9
|
+
- [Exchanges] Handle order type open status for symbol
|
10
|
+
### Updated
|
11
|
+
- [Exchanges] Handle portfolio optimization using limit orders
|
12
|
+
|
13
|
+
## [2.4.75] - 2024-04-11
|
14
|
+
### Fixed
|
15
|
+
- [Exchanges] Properly handle order not found errors
|
16
|
+
- [Order] Rare synch issues on creation
|
17
|
+
- [Order] Missing ungrouped stop orders when restarting
|
18
|
+
|
7
19
|
## [2.4.74] - 2024-04-07
|
8
20
|
### Added
|
9
21
|
- [Exchanges] ExchangeCompliancyError error
|
@@ -1,6 +1,6 @@
|
|
1
1
|
Metadata-Version: 2.1
|
2
2
|
Name: OctoBot-Trading
|
3
|
-
Version: 2.4.
|
3
|
+
Version: 2.4.76
|
4
4
|
Summary: OctoBot project trading package
|
5
5
|
Home-page: https://github.com/Drakkar-Software/OctoBot-Trading
|
6
6
|
Author: Drakkar-Software
|
@@ -16,7 +16,7 @@ Classifier: Operating System :: POSIX
|
|
16
16
|
Requires-Python: >=3.8
|
17
17
|
License-File: LICENSE
|
18
18
|
|
19
|
-
# OctoBot-Trading [2.4.
|
19
|
+
# OctoBot-Trading [2.4.76](https://github.com/Drakkar-Software/OctoBot-Trading/blob/master/CHANGELOG.md)
|
20
20
|
[](https://app.codacy.com/gh/Drakkar-Software/OctoBot-Trading?utm_source=github.com&utm_medium=referral&utm_content=Drakkar-Software/OctoBot-Trading&utm_campaign=Badge_Grade_Dashboard)
|
21
21
|
[](https://pypi.python.org/pypi/OctoBot-Trading/)
|
22
22
|
[](https://coveralls.io/github/Drakkar-Software/OctoBot-Trading?branch=master)
|
@@ -1,6 +1,6 @@
|
|
1
1
|
Metadata-Version: 2.1
|
2
2
|
Name: OctoBot-Trading
|
3
|
-
Version: 2.4.
|
3
|
+
Version: 2.4.76
|
4
4
|
Summary: OctoBot project trading package
|
5
5
|
Home-page: https://github.com/Drakkar-Software/OctoBot-Trading
|
6
6
|
Author: Drakkar-Software
|
@@ -16,7 +16,7 @@ Classifier: Operating System :: POSIX
|
|
16
16
|
Requires-Python: >=3.8
|
17
17
|
License-File: LICENSE
|
18
18
|
|
19
|
-
# OctoBot-Trading [2.4.
|
19
|
+
# OctoBot-Trading [2.4.76](https://github.com/Drakkar-Software/OctoBot-Trading/blob/master/CHANGELOG.md)
|
20
20
|
[](https://app.codacy.com/gh/Drakkar-Software/OctoBot-Trading?utm_source=github.com&utm_medium=referral&utm_content=Drakkar-Software/OctoBot-Trading&utm_campaign=Badge_Grade_Dashboard)
|
21
21
|
[](https://pypi.python.org/pypi/OctoBot-Trading/)
|
22
22
|
[](https://coveralls.io/github/Drakkar-Software/OctoBot-Trading?branch=master)
|
@@ -1,4 +1,4 @@
|
|
1
|
-
# OctoBot-Trading [2.4.
|
1
|
+
# OctoBot-Trading [2.4.76](https://github.com/Drakkar-Software/OctoBot-Trading/blob/master/CHANGELOG.md)
|
2
2
|
[](https://app.codacy.com/gh/Drakkar-Software/OctoBot-Trading?utm_source=github.com&utm_medium=referral&utm_content=Drakkar-Software/OctoBot-Trading&utm_campaign=Badge_Grade_Dashboard)
|
3
3
|
[](https://pypi.python.org/pypi/OctoBot-Trading/)
|
4
4
|
[](https://coveralls.io/github/Drakkar-Software/OctoBot-Trading?branch=master)
|
@@ -42,6 +42,9 @@ ALLOW_SIMULATED_ORDERS_INSTANT_FILL = os_util.parse_boolean_environment_var(
|
|
42
42
|
ORDER_DATA_FETCHING_TIMEOUT = 5 * commons_constants.MINUTE_TO_SECONDS
|
43
43
|
CHAINED_ORDER_PRICE_FETCHING_TIMEOUT = 1 # should be instant or ignored
|
44
44
|
CHAINED_ORDERS_OUTDATED_PRICE_ALLOWANCE = decimal.Decimal("0.005") # allows 0.5% outdated price error
|
45
|
+
# create instantly filled limit orders 0.5% beyond market
|
46
|
+
INSTANT_FILLED_LIMIT_ORDER_PRICE_DELTA = decimal.Decimal("0.005")
|
47
|
+
CREATED_ORDER_FORCED_UPDATE_PERIOD = 5
|
45
48
|
|
46
49
|
# Tentacles
|
47
50
|
TRADING_MODE_REQUIRED_STRATEGIES = "required_strategies"
|
@@ -130,13 +133,13 @@ EXCHANGE_PERMISSION_ERRORS: typing.List[typing.Iterable[str]] = [
|
|
130
133
|
|
131
134
|
# coinbase ex: coinbase {"error":"PERMISSION_DENIED",
|
132
135
|
# "error_details":"Missing required scopes","message":"Missing required scopes"}
|
133
|
-
("permission_denied", )
|
136
|
+
("permission_denied", "required scopes"),
|
134
137
|
]
|
135
138
|
|
136
139
|
# text content of errors due to exchange compliancy rules
|
137
140
|
EXCHANGE_COMPLIANCY_ERRORS: typing.List[typing.Iterable[str]] = [
|
138
141
|
# OKX ex: Trading of this pair or contract is restricted due to local compliance requirements
|
139
|
-
("restricted
|
142
|
+
("restricted", "compliance"),
|
140
143
|
]
|
141
144
|
|
142
145
|
CONFIG_DEFAULT_FEES = 0.001
|
@@ -93,6 +93,12 @@ class FailedRequest(Exception):
|
|
93
93
|
"""
|
94
94
|
|
95
95
|
|
96
|
+
class UnavailableOrderTypeForMarketError(Exception):
|
97
|
+
"""
|
98
|
+
Raised when an exchange refuses to create a given type of order that should normally be supported
|
99
|
+
"""
|
100
|
+
|
101
|
+
|
96
102
|
class AuthenticationError(Exception):
|
97
103
|
"""
|
98
104
|
Raised when an exchange failed to authenticate
|
@@ -78,6 +78,7 @@ from octobot_trading.exchanges.util import (
|
|
78
78
|
get_exchange_details,
|
79
79
|
is_api_permission_error,
|
80
80
|
is_exchange_rules_compliancy_error,
|
81
|
+
is_error_on_this_type,
|
81
82
|
force_disable_web_socket,
|
82
83
|
check_web_socket_config,
|
83
84
|
search_websocket_class,
|
@@ -164,6 +165,7 @@ __all__ = [
|
|
164
165
|
"get_exchange_details",
|
165
166
|
"is_api_permission_error",
|
166
167
|
"is_exchange_rules_compliancy_error",
|
168
|
+
"is_error_on_this_type",
|
167
169
|
"AbstractExchange",
|
168
170
|
"is_channel_managed_by_websocket",
|
169
171
|
"is_channel_fully_managed_by_websocket",
|
{OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/abstract_exchange.py
RENAMED
@@ -17,6 +17,7 @@ import asyncio
|
|
17
17
|
import typing
|
18
18
|
import decimal
|
19
19
|
import time
|
20
|
+
import contextlib
|
20
21
|
|
21
22
|
import octobot_commons.constants
|
22
23
|
import octobot_commons.enums as common_enums
|
@@ -94,6 +95,8 @@ class AbstractExchange(tentacles_management.AbstractTentacle):
|
|
94
95
|
|
95
96
|
self.is_unreachable = False
|
96
97
|
|
98
|
+
self._creating_exchange_order_ids = set()
|
99
|
+
|
97
100
|
if self.exchange_manager.tentacles_setup_config is not None:
|
98
101
|
self.load_user_inputs_from_class(self.exchange_manager.tentacles_setup_config, self.tentacle_config)
|
99
102
|
|
@@ -380,7 +383,7 @@ class AbstractExchange(tentacles_management.AbstractTentacle):
|
|
380
383
|
"""
|
381
384
|
raise NotImplementedError("get_bundled_order_parameters is not implemented")
|
382
385
|
|
383
|
-
def is_supported_order_type(self, order_type):
|
386
|
+
def is_supported_order_type(self, order_type: enums.TraderOrderType) -> bool:
|
384
387
|
"""
|
385
388
|
Check if the order type is supported by the current exchange instance
|
386
389
|
Should be used to know if we should simulate this order or create it on the exchange
|
@@ -389,6 +392,12 @@ class AbstractExchange(tentacles_management.AbstractTentacle):
|
|
389
392
|
"""
|
390
393
|
return order_type not in self.get_supported_elements(enums.ExchangeSupportedElements.UNSUPPORTED_ORDERS)
|
391
394
|
|
395
|
+
def is_market_open_for_order_type(self, symbol: str, order_type: enums.TraderOrderType) -> bool:
|
396
|
+
"""
|
397
|
+
Override if necessary
|
398
|
+
"""
|
399
|
+
return True
|
400
|
+
|
392
401
|
def get_trade_fee(self, symbol: str, order_type: enums.TraderOrderType, quantity, price, taker_or_maker):
|
393
402
|
"""
|
394
403
|
Calculates fees resulting to a trade
|
@@ -698,3 +707,19 @@ class AbstractExchange(tentacles_management.AbstractTentacle):
|
|
698
707
|
def handle_token_error(self, error):
|
699
708
|
self.logger.error(f"Exchange configuration is invalid : please check your configuration ! "
|
700
709
|
f"({error.__class__.__name__}: {error})")
|
710
|
+
|
711
|
+
@contextlib.contextmanager
|
712
|
+
def creating_order(self, creating_order: dict):
|
713
|
+
exchange_order_id = creating_order.get(
|
714
|
+
enums.ExchangeConstantsOrderColumns.EXCHANGE_ID.value
|
715
|
+
) if creating_order else None
|
716
|
+
try:
|
717
|
+
self._creating_exchange_order_ids.add(exchange_order_id)
|
718
|
+
yield
|
719
|
+
finally:
|
720
|
+
self._creating_exchange_order_ids.remove(exchange_order_id)
|
721
|
+
|
722
|
+
def is_creating_order(self, exchange_order_id):
|
723
|
+
if exchange_order_id is None:
|
724
|
+
return False
|
725
|
+
return exchange_order_id in self._creating_exchange_order_ids
|
@@ -351,8 +351,12 @@ class CCXTConnector(abstract_exchange.AbstractExchange):
|
|
351
351
|
# some exchanges are throwing this error when an order is cancelled (ex: coinbase pro)
|
352
352
|
raise octobot_trading.errors.NotSupported(e) from e
|
353
353
|
except ccxt.ExchangeError as e:
|
354
|
-
|
355
|
-
|
354
|
+
if self.exchange_manager.exchange.is_order_not_found_error(e):
|
355
|
+
# when an OrderNotFound error should have been raised but is not for some reason
|
356
|
+
pass
|
357
|
+
else:
|
358
|
+
# something went wrong and ccxt did not expect it
|
359
|
+
raise octobot_trading.errors.FailedRequest(e) from e
|
356
360
|
else:
|
357
361
|
# When fetch_order is not supported, uses get_open_orders or get_closed_orders and extract order id
|
358
362
|
for method in (self.get_open_orders, self.get_closed_orders):
|
{OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/traders/trader.py
RENAMED
@@ -279,7 +279,9 @@ class Trader(util.Initializable):
|
|
279
279
|
|
280
280
|
if is_pending_creation:
|
281
281
|
# register order as pending order, it will then be added to live orders in order manager once open
|
282
|
-
self.exchange_manager.exchange_personal_data.orders_manager.register_pending_creation_order(
|
282
|
+
self.exchange_manager.exchange_personal_data.orders_manager.register_pending_creation_order(
|
283
|
+
updated_order
|
284
|
+
)
|
283
285
|
|
284
286
|
await updated_order.initialize()
|
285
287
|
if is_pending_creation and wait_for_creation \
|
{OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/types/rest_exchange.py
RENAMED
@@ -89,6 +89,9 @@ class RestExchange(abstract_exchange.AbstractExchange):
|
|
89
89
|
# Set when order cost is not (yet) accurately computed for a given exchange
|
90
90
|
MAX_INCREASED_POSITION_QUANTITY_MULTIPLIER = constants.ONE
|
91
91
|
|
92
|
+
# text content of errors due to orders not found errors
|
93
|
+
EXCHANGE_ORDER_NOT_FOUND_ERRORS: typing.List[typing.Iterable[str]] = []
|
94
|
+
|
92
95
|
DEFAULT_CONNECTOR_CLASS = ccxt_connector.CCXTConnector
|
93
96
|
|
94
97
|
def __init__(self, config, exchange_manager, connector_class=None):
|
@@ -152,7 +155,8 @@ class RestExchange(abstract_exchange.AbstractExchange):
|
|
152
155
|
stop_price=stop_price, side=side, current_price=current_price,
|
153
156
|
reduce_only=reduce_only, params=params)
|
154
157
|
self.logger.debug(f"Created order: {created_order}")
|
155
|
-
|
158
|
+
with self.creating_order(created_order):
|
159
|
+
return await self._verify_order(created_order, order_type, symbol, price, side)
|
156
160
|
return None
|
157
161
|
|
158
162
|
async def edit_order(self, exchange_order_id: str, order_type: enums.TraderOrderType, symbol: str,
|
@@ -192,30 +196,35 @@ class RestExchange(abstract_exchange.AbstractExchange):
|
|
192
196
|
self.log_order_creation_error(e, order_type, symbol, quantity, price, stop_price)
|
193
197
|
if self.__class__.PRINT_DEBUG_LOGS:
|
194
198
|
self.logger.warning(str(e))
|
195
|
-
raise errors.MissingFunds(e)
|
196
|
-
except ccxt.NotSupported:
|
197
|
-
raise errors.NotSupported
|
199
|
+
raise errors.MissingFunds(e) from e
|
200
|
+
except ccxt.NotSupported as err:
|
201
|
+
raise errors.NotSupported from err
|
198
202
|
except ccxt.AuthenticationError as err:
|
199
203
|
# invalid api key or missing trading rights
|
200
204
|
raise errors.AuthenticationError(
|
201
205
|
f"Error when handling order {err}. Please make sure that trading permissions are on for this API key."
|
202
|
-
)
|
206
|
+
) from err
|
203
207
|
except ccxt.DDoSProtection as e:
|
204
208
|
# raised upon rate limit issues, last response data might have details on what is happening
|
205
209
|
if self.should_log_on_ddos_exception(e):
|
206
210
|
self.connector.log_ddos_error(e)
|
207
211
|
raise errors.FailedRequest(f"Failed to order operation: {e.__class__.__name__} {e}") from e
|
208
212
|
except Exception as e:
|
213
|
+
if not self.is_market_open_for_order_type(symbol, order_type):
|
214
|
+
raise errors.UnavailableOrderTypeForMarketError(
|
215
|
+
f"Error when handling order {e}. "
|
216
|
+
f"Exchange currently refuses to create orders of type {order_type} on {symbol}."
|
217
|
+
) from e
|
209
218
|
if exchanges_util.is_api_permission_error(e):
|
210
219
|
# invalid api key or missing trading rights
|
211
220
|
raise errors.AuthenticationError(
|
212
221
|
f"Error when handling order {e}. Please make sure that trading permissions are on for this API key."
|
213
|
-
)
|
222
|
+
) from e
|
214
223
|
if exchanges_util.is_exchange_rules_compliancy_error(e):
|
215
224
|
raise errors.ExchangeCompliancyError(
|
216
225
|
f"Error when handling order {e}. Exchange is refusing this order request on this account because "
|
217
226
|
f"of its compliancy requirements."
|
218
|
-
)
|
227
|
+
) from e
|
219
228
|
self.log_order_creation_error(e, order_type, symbol, quantity, price, stop_price)
|
220
229
|
print(traceback.format_exc(), file=sys.stderr)
|
221
230
|
self.logger.exception(e, False, f"Unexpected error during order operation: {e}")
|
@@ -866,6 +875,11 @@ class RestExchange(abstract_exchange.AbstractExchange):
|
|
866
875
|
def is_skipping_empty_candles_in_ohlcv_fetch(self):
|
867
876
|
return self.IS_SKIPPING_EMPTY_CANDLES_IN_OHLCV_FETCH
|
868
877
|
|
878
|
+
def is_order_not_found_error(self, error: BaseException) -> bool:
|
879
|
+
if self.EXCHANGE_ORDER_NOT_FOUND_ERRORS:
|
880
|
+
return exchanges_util.is_error_on_this_type(error, self.EXCHANGE_ORDER_NOT_FOUND_ERRORS)
|
881
|
+
return False
|
882
|
+
|
869
883
|
"""
|
870
884
|
Auto fetched and filled exchanges
|
871
885
|
"""
|
{OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/util/__init__.py
RENAMED
@@ -46,6 +46,7 @@ from octobot_trading.exchanges.util.exchange_util import (
|
|
46
46
|
get_exchange_details,
|
47
47
|
is_api_permission_error,
|
48
48
|
is_exchange_rules_compliancy_error,
|
49
|
+
is_error_on_this_type,
|
49
50
|
)
|
50
51
|
from octobot_trading.exchanges.util import websockets_util
|
51
52
|
from octobot_trading.exchanges.util.websockets_util import (
|
@@ -80,6 +81,7 @@ __all__ = [
|
|
80
81
|
"get_exchange_details",
|
81
82
|
"is_api_permission_error",
|
82
83
|
"is_exchange_rules_compliancy_error",
|
84
|
+
"is_error_on_this_type",
|
83
85
|
"force_disable_web_socket",
|
84
86
|
"check_web_socket_config",
|
85
87
|
"search_websocket_class",
|
{OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/octobot_trading/exchanges/util/exchange_util.py
RENAMED
@@ -429,14 +429,14 @@ def get_associated_symbol(exchange_manager, asset: str, target_asset: str) -> (t
|
|
429
429
|
|
430
430
|
|
431
431
|
def is_api_permission_error(error: BaseException) -> bool:
|
432
|
-
return
|
432
|
+
return is_error_on_this_type(error, constants.EXCHANGE_PERMISSION_ERRORS)
|
433
433
|
|
434
434
|
|
435
435
|
def is_exchange_rules_compliancy_error(error: BaseException) -> bool:
|
436
|
-
return
|
436
|
+
return is_error_on_this_type(error, constants.EXCHANGE_COMPLIANCY_ERRORS)
|
437
437
|
|
438
438
|
|
439
|
-
def
|
439
|
+
def is_error_on_this_type(error: BaseException, descriptions: typing.List[typing.Iterable[str]]) -> bool:
|
440
440
|
lower_error = str(error).lower()
|
441
441
|
for identifiers in descriptions:
|
442
442
|
if all(identifier in lower_error for identifier in identifiers):
|
@@ -120,6 +120,18 @@ async def convert_asset_to_target_asset(
|
|
120
120
|
)
|
121
121
|
return created_orders
|
122
122
|
|
123
|
+
if trading_personal_data.get_trade_order_type(order_type) is not trading_enums.TradeOrderType.MARKET:
|
124
|
+
# can't use market orders: use limit orders with price a bit under the current price to instant fill it.
|
125
|
+
price_delta = price * constants.INSTANT_FILLED_LIMIT_ORDER_PRICE_DELTA
|
126
|
+
if order_type is trading_enums.TraderOrderType.SELL_LIMIT:
|
127
|
+
price -= price_delta
|
128
|
+
elif order_type is trading_enums.TraderOrderType.BUY_LIMIT:
|
129
|
+
price += price_delta
|
130
|
+
else:
|
131
|
+
trading_mode.logger.error(
|
132
|
+
f"Unhandled order type in convertor limit order price adapter: {order_type}"
|
133
|
+
)
|
134
|
+
|
123
135
|
# get order quantity
|
124
136
|
quantity = _get_available_or_target_quantity(trading_mode, symbol, order_type, price, asset_amount)
|
125
137
|
symbol_market = trading_mode.exchange_manager.exchange.get_market_status(symbol, with_fixer=False)
|
@@ -147,6 +159,15 @@ def _get_associated_symbol_and_order_type(trading_mode, asset: str, target_asset
|
|
147
159
|
symbol, reversed_symbol = exchange_util.get_associated_symbol(trading_mode.exchange_manager, asset, target_asset)
|
148
160
|
order_type = trading_enums.TraderOrderType.BUY_MARKET if reversed_symbol else \
|
149
161
|
trading_enums.TraderOrderType.SELL_MARKET
|
162
|
+
if not trading_mode.exchange_manager.exchange.is_market_open_for_order_type(symbol, order_type):
|
163
|
+
# can't use market orders: use limit orders instead
|
164
|
+
order_type = trading_enums.TraderOrderType.BUY_LIMIT if order_type is trading_enums.TraderOrderType.BUY_MARKET \
|
165
|
+
else trading_enums.TraderOrderType.SELL_LIMIT
|
166
|
+
if not trading_mode.exchange_manager.exchange.is_market_open_for_order_type(symbol, order_type):
|
167
|
+
# can't convert asset: still try with limit orders but it will probably fail
|
168
|
+
trading_mode.logger.error(
|
169
|
+
f"Both market and {order_type} order are currently unsupported. Trying limit orders anyway."
|
170
|
+
)
|
150
171
|
return symbol, order_type
|
151
172
|
|
152
173
|
|
@@ -36,11 +36,19 @@ class OrdersProducer(exchanges_channel.ExchangeChannelProducer):
|
|
36
36
|
pending_groups = {} # Used when restoring orders from order storage:
|
37
37
|
# a dict of order groups for which to check if associated self-managed orders are to be created
|
38
38
|
for order in orders:
|
39
|
+
exchange_order_id: str = self.channel.exchange_manager.exchange.parse_exhange_order_id(order)
|
39
40
|
symbol = self.channel.exchange_manager.get_exchange_symbol(
|
40
41
|
self.channel.exchange_manager.exchange.parse_order_symbol(order)
|
41
42
|
)
|
43
|
+
if self.channel.exchange_manager.exchange.is_creating_order(exchange_order_id):
|
44
|
+
# ignore orders that are being created
|
45
|
+
self.logger.debug(
|
46
|
+
f"Ignored update from order channel for {symbol} order with exchange order id "
|
47
|
+
f"{exchange_order_id} as "
|
48
|
+
f"this order is being created and will automatically be updated once creation is complete."
|
49
|
+
)
|
50
|
+
continue
|
42
51
|
symbols.add(symbol)
|
43
|
-
exchange_order_id: str = self.channel.exchange_manager.exchange.parse_exhange_order_id(order)
|
44
52
|
|
45
53
|
# if this order was not managed by order_manager before
|
46
54
|
is_new_order = not self.channel.exchange_manager.exchange_personal_data.orders_manager. \
|
@@ -83,6 +91,16 @@ class OrdersProducer(exchanges_channel.ExchangeChannelProducer):
|
|
83
91
|
except Exception as e:
|
84
92
|
self.logger.exception(e, True, f"Exception when triggering update: {e}")
|
85
93
|
|
94
|
+
async def _restore_required_virtual_orders(self):
|
95
|
+
"""
|
96
|
+
Restore virtual orders that would not be restored otherwise
|
97
|
+
Should only be called once or will create the same virtual orders multiple times
|
98
|
+
"""
|
99
|
+
pending_groups = {}
|
100
|
+
await orders_storage_operations.create_required_virtual_orders(
|
101
|
+
pending_groups, self.channel.exchange_manager
|
102
|
+
)
|
103
|
+
|
86
104
|
async def _handle_open_order_update(
|
87
105
|
self, symbol, order_dict, exchange_order_id, is_from_bot, is_new_order, pending_groups
|
88
106
|
):
|
@@ -76,6 +76,7 @@ class OrdersUpdater(orders_channel.OrdersProducer):
|
|
76
76
|
self.DEPENDENCIES_TIMEOUT
|
77
77
|
)
|
78
78
|
await self.fetch_and_push(is_from_bot=False, retry_till_success=True)
|
79
|
+
await self._restore_required_virtual_orders()
|
79
80
|
except errors.NotSupported:
|
80
81
|
self.logger.error(f"{self.channel.exchange_manager.exchange_name} is not supporting open orders updates")
|
81
82
|
await self.pause()
|
@@ -13,6 +13,8 @@
|
|
13
13
|
#
|
14
14
|
# You should have received a copy of the GNU Lesser General Public
|
15
15
|
# License along with this library.
|
16
|
+
import octobot_commons.logging as logging
|
17
|
+
|
16
18
|
import octobot_trading.personal_data as personal_data
|
17
19
|
import octobot_trading.enums as enums
|
18
20
|
import octobot_trading.constants as constants
|
@@ -150,3 +152,4 @@ async def restore_chained_orders_from_storage_order_details(order, order_details
|
|
150
152
|
**chained_order.get(enums.StoredOrdersAttr.TRADER_CREATION_KWARGS.value, {}),
|
151
153
|
)
|
152
154
|
order.add_chained_order(chained_order_inst)
|
155
|
+
logging.get_logger(order.get_logger_name()).debug(f"Restored chained order: {chained_order_inst}")
|
@@ -14,6 +14,7 @@
|
|
14
14
|
# You should have received a copy of the GNU Lesser General Public
|
15
15
|
# License along with this library.
|
16
16
|
import octobot_commons.logging as logging
|
17
|
+
import octobot_trading.enums as enums
|
17
18
|
import octobot_trading.personal_data.orders.order_factory as order_factory
|
18
19
|
import octobot_trading.personal_data.orders.groups.group_util as group_util
|
19
20
|
|
@@ -30,6 +31,7 @@ async def apply_order_storage_details_if_any(order, exchange_manager, pending_gr
|
|
30
31
|
order.exchange_order_id
|
31
32
|
)
|
32
33
|
if order_details:
|
34
|
+
logging.get_logger(LOGGER_NAME).debug(f"Updating fetched order {order} using stored order details")
|
33
35
|
order.update_from_storage_order_details(order_details)
|
34
36
|
await create_orders_storage_related_elements(order, order_details, exchange_manager, pending_groups)
|
35
37
|
|
@@ -45,21 +47,43 @@ async def create_orders_storage_related_elements(order, order_storage_details, e
|
|
45
47
|
)
|
46
48
|
|
47
49
|
|
50
|
+
async def create_order_from_storage_data(order_desc: dict, exchange_manager, pending_groups):
|
51
|
+
created_order = await order_factory.create_order_from_order_storage_details(
|
52
|
+
order_desc, exchange_manager, pending_groups
|
53
|
+
)
|
54
|
+
await created_order.initialize()
|
55
|
+
return created_order
|
56
|
+
|
57
|
+
|
48
58
|
async def _create_storage_self_managed_orders_from_group(pending_group_id, exchange_manager, pending_groups):
|
49
59
|
try:
|
50
60
|
to_create_orders = exchange_manager.storage_manager.orders_storage \
|
51
61
|
.get_startup_self_managed_orders_details_from_group(pending_group_id)
|
52
62
|
for order_desc in to_create_orders:
|
53
|
-
|
54
|
-
order_desc, exchange_manager, pending_groups
|
55
|
-
)
|
56
|
-
await created_order.initialize()
|
63
|
+
await create_order_from_storage_data(order_desc, exchange_manager, pending_groups)
|
57
64
|
except Exception as err:
|
58
65
|
logging.get_logger(LOGGER_NAME).exception(
|
59
66
|
err, True, f"Error when creating {pending_group_id} group self-managed orders with stored data: {err}"
|
60
67
|
)
|
61
68
|
|
62
69
|
|
70
|
+
async def create_required_virtual_orders(pending_groups, exchange_manager):
|
71
|
+
virtual_orders = exchange_manager.storage_manager.orders_storage.get_all_self_managed_startup_orders()
|
72
|
+
# virtual order that are not within a group will only be restored here
|
73
|
+
virtual_non_grouped_order_details = [
|
74
|
+
order
|
75
|
+
for order in virtual_orders
|
76
|
+
if order.get(enums.StoredOrdersAttr.GROUP.value, {}) == {}
|
77
|
+
]
|
78
|
+
logger = logging.get_logger(LOGGER_NAME)
|
79
|
+
for order_details in virtual_non_grouped_order_details:
|
80
|
+
try:
|
81
|
+
order = await create_order_from_storage_data(order_details, exchange_manager, pending_groups)
|
82
|
+
logger.debug(f"Restored order from storage: {order}")
|
83
|
+
except Exception as err:
|
84
|
+
logger.exception(err, True, f"Error when restoring virtual order order using stored data: {err}")
|
85
|
+
|
86
|
+
|
63
87
|
async def create_missing_self_managed_orders_from_storage_order_groups(pending_groups, exchange_manager):
|
64
88
|
# create order groups' associated self-managed orders if any
|
65
89
|
to_complete_groups = list(pending_groups.keys())
|
@@ -124,10 +124,16 @@ class OrdersStorage(abstract_storage.AbstractStorage):
|
|
124
124
|
def get_startup_self_managed_orders_details_from_group(self, group_id):
|
125
125
|
return [
|
126
126
|
order
|
127
|
-
for order in self.
|
127
|
+
for order in self.get_all_self_managed_startup_orders()
|
128
128
|
if order.get(enums.StoredOrdersAttr.GROUP.value, {}).get(enums.StoredOrdersAttr.GROUP_ID.value, None)
|
129
129
|
== group_id
|
130
|
-
|
130
|
+
]
|
131
|
+
|
132
|
+
def get_all_self_managed_startup_orders(self):
|
133
|
+
return [
|
134
|
+
order
|
135
|
+
for order in self.startup_orders.values()
|
136
|
+
if order.get(constants.STORAGE_ORIGIN_VALUE, {})
|
131
137
|
.get(enums.ExchangeConstantsOrderColumns.SELF_MANAGED.value, False)
|
132
138
|
]
|
133
139
|
|
@@ -21,6 +21,8 @@ import typing
|
|
21
21
|
import octobot_commons.asyncio_tools as asyncio_tools
|
22
22
|
import octobot_commons.enums as common_enums
|
23
23
|
import octobot_commons.constants as commons_constants
|
24
|
+
import octobot_commons.logging as logging
|
25
|
+
import octobot_trading.constants as constants
|
24
26
|
import octobot_trading.enums as enums
|
25
27
|
import octobot_trading.api as trading_api
|
26
28
|
import octobot_trading.exchanges as exchanges
|
@@ -172,7 +174,12 @@ async def get_trades(
|
|
172
174
|
return trades
|
173
175
|
|
174
176
|
|
175
|
-
async def create_orders(
|
177
|
+
async def create_orders(
|
178
|
+
exchange_manager,
|
179
|
+
exchange_data: exchange_data_import.ExchangeData,
|
180
|
+
orders: list,
|
181
|
+
order_creation_timeout: float
|
182
|
+
) -> list:
|
176
183
|
async def _create_order(order_dict) -> personal_data.Order:
|
177
184
|
symbol = order_dict[enums.ExchangeConstantsOrderColumns.SYMBOL.value]
|
178
185
|
side, order_type = personal_data.parse_order_type(order_dict)
|
@@ -186,8 +193,36 @@ async def create_orders(exchange_manager, exchange_data: exchange_data_import.Ex
|
|
186
193
|
reduce_only=order_dict[enums.ExchangeConstantsOrderColumns.REDUCE_ONLY.value],
|
187
194
|
)
|
188
195
|
# is private, to use in tests context only
|
189
|
-
|
196
|
+
order = personal_data.create_order_instance_from_raw(
|
190
197
|
exchange_manager.trader, created_order, force_open_or_pending_creation=True
|
191
198
|
) if created_order else None
|
199
|
+
if not order:
|
200
|
+
return order
|
201
|
+
if order.status is enums.OrderStatus.PENDING_CREATION and order_creation_timeout > 0:
|
202
|
+
try:
|
203
|
+
return await wait_for_other_status(order, order_creation_timeout)
|
204
|
+
except TimeoutError as err:
|
205
|
+
logging.get_logger(order.get_logger_name()).error(f"Created order can't be fetched: {err}")
|
206
|
+
return None
|
207
|
+
return order
|
192
208
|
|
193
209
|
return await asyncio.gather(*(_create_order(order_dict) for order_dict in orders))
|
210
|
+
|
211
|
+
|
212
|
+
async def wait_for_other_status(order: personal_data.Order, timeout) -> personal_data.Order:
|
213
|
+
t0 = time.time()
|
214
|
+
iterations = 0
|
215
|
+
origin_status = order.status.value
|
216
|
+
while time.time() - t0 < timeout:
|
217
|
+
raw_order = await order.exchange_manager.exchange.get_order(order.exchange_order_id, order.symbol)
|
218
|
+
iterations += 1
|
219
|
+
if raw_order is not None and raw_order[enums.ExchangeConstantsOrderColumns.STATUS.value] != origin_status:
|
220
|
+
logging.get_logger(order.get_logger_name()).info(
|
221
|
+
f"Order fetched with status different from {origin_status} after {iterations} "
|
222
|
+
f"iterations and {round(time.time() - t0)}s"
|
223
|
+
)
|
224
|
+
return personal_data.create_order_instance_from_raw(
|
225
|
+
order.trader, raw_order, force_open_or_pending_creation=False
|
226
|
+
)
|
227
|
+
await asyncio.sleep(constants.CREATED_ORDER_FORCED_UPDATE_PERIOD)
|
228
|
+
raise TimeoutError(f"Order was not found with another status than {origin_status} within {timeout} seconds")
|
{OctoBot-Trading-2.4.74 → OctoBot-Trading-2.4.76}/tests/exchanges/util/test_exchange_util.py
RENAMED
@@ -245,25 +245,25 @@ async def test_get_exchange_details(tentacles_setup_config, supported_exchanges)
|
|
245
245
|
|
246
246
|
|
247
247
|
def test_is_api_permission_error():
|
248
|
-
with mock.patch.object(exchange_util, "
|
248
|
+
with mock.patch.object(exchange_util, "is_error_on_this_type") as is_error_on_this_type_mock:
|
249
249
|
err = Exception("plop")
|
250
250
|
exchanges.is_api_permission_error(err)
|
251
|
-
|
251
|
+
is_error_on_this_type_mock.assert_called_once_with(
|
252
252
|
err, constants.EXCHANGE_PERMISSION_ERRORS
|
253
253
|
)
|
254
254
|
|
255
255
|
|
256
256
|
def test_is_exchange_rules_compliancy_error():
|
257
|
-
with mock.patch.object(exchange_util, "
|
257
|
+
with mock.patch.object(exchange_util, "is_error_on_this_type") as is_error_on_this_type_mock:
|
258
258
|
err = Exception("plop")
|
259
259
|
exchanges.is_exchange_rules_compliancy_error(err)
|
260
|
-
|
260
|
+
is_error_on_this_type_mock.assert_called_once_with(
|
261
261
|
err, constants.EXCHANGE_COMPLIANCY_ERRORS
|
262
262
|
)
|
263
263
|
|
264
264
|
|
265
265
|
def test_is_error_on_this_type():
|
266
|
-
assert exchange_util.
|
267
|
-
assert exchange_util.
|
268
|
-
assert exchange_util.
|
269
|
-
assert exchange_util.
|
266
|
+
assert exchange_util.is_error_on_this_type(Exception("plop"), constants.EXCHANGE_PERMISSION_ERRORS) is False
|
267
|
+
assert exchange_util.is_error_on_this_type(Exception("api key"), constants.EXCHANGE_PERMISSION_ERRORS) is True
|
268
|
+
assert exchange_util.is_error_on_this_type(Exception("api"), constants.EXCHANGE_PERMISSION_ERRORS) is False
|
269
|
+
assert exchange_util.is_error_on_this_type(Exception("api"), constants.EXCHANGE_COMPLIANCY_ERRORS) is False
|