Merge branch 'develop' of https://github.com/Surfableio/freqtrade into develop
This commit is contained in:
		
							
								
								
									
										28
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										28
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							| @@ -13,6 +13,10 @@ on: | |||||||
|   schedule: |   schedule: | ||||||
|     - cron:  '0 5 * * 4' |     - cron:  '0 5 * * 4' | ||||||
|  |  | ||||||
|  | concurrency: | ||||||
|  |   group: ${{ github.workflow }}-${{ github.ref }} | ||||||
|  |   cancel-in-progress: true | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   build_linux: |   build_linux: | ||||||
|  |  | ||||||
| @@ -26,7 +30,7 @@ jobs: | |||||||
|     - uses: actions/checkout@v3 |     - uses: actions/checkout@v3 | ||||||
|  |  | ||||||
|     - name: Set up Python |     - name: Set up Python | ||||||
|       uses: actions/setup-python@v3 |       uses: actions/setup-python@v4 | ||||||
|       with: |       with: | ||||||
|         python-version: ${{ matrix.python-version }} |         python-version: ${{ matrix.python-version }} | ||||||
|  |  | ||||||
| @@ -123,7 +127,7 @@ jobs: | |||||||
|     - uses: actions/checkout@v3 |     - uses: actions/checkout@v3 | ||||||
|  |  | ||||||
|     - name: Set up Python |     - name: Set up Python | ||||||
|       uses: actions/setup-python@v3 |       uses: actions/setup-python@v4 | ||||||
|       with: |       with: | ||||||
|         python-version: ${{ matrix.python-version }} |         python-version: ${{ matrix.python-version }} | ||||||
|  |  | ||||||
| @@ -207,7 +211,7 @@ jobs: | |||||||
|     - uses: actions/checkout@v3 |     - uses: actions/checkout@v3 | ||||||
|  |  | ||||||
|     - name: Set up Python |     - name: Set up Python | ||||||
|       uses: actions/setup-python@v3 |       uses: actions/setup-python@v4 | ||||||
|       with: |       with: | ||||||
|         python-version: ${{ matrix.python-version }} |         python-version: ${{ matrix.python-version }} | ||||||
|  |  | ||||||
| @@ -259,7 +263,7 @@ jobs: | |||||||
|     - uses: actions/checkout@v3 |     - uses: actions/checkout@v3 | ||||||
|  |  | ||||||
|     - name: Set up Python |     - name: Set up Python | ||||||
|       uses: actions/setup-python@v3 |       uses: actions/setup-python@v4 | ||||||
|       with: |       with: | ||||||
|         python-version: "3.10" |         python-version: "3.10" | ||||||
|  |  | ||||||
| @@ -278,7 +282,7 @@ jobs: | |||||||
|         ./tests/test_docs.sh |         ./tests/test_docs.sh | ||||||
|  |  | ||||||
|     - name: Set up Python |     - name: Set up Python | ||||||
|       uses: actions/setup-python@v3 |       uses: actions/setup-python@v4 | ||||||
|       with: |       with: | ||||||
|         python-version: "3.10" |         python-version: "3.10" | ||||||
|  |  | ||||||
| @@ -296,18 +300,6 @@ jobs: | |||||||
|           details: Freqtrade doc test failed! |           details: Freqtrade doc test failed! | ||||||
|           webhookUrl: ${{ secrets.DISCORD_WEBHOOK }} |           webhookUrl: ${{ secrets.DISCORD_WEBHOOK }} | ||||||
|  |  | ||||||
|   cleanup-prior-runs: |  | ||||||
|     permissions: |  | ||||||
|       actions: write  # for rokroskar/workflow-run-cleanup-action to obtain workflow name & cancel it |  | ||||||
|       contents: read  # for rokroskar/workflow-run-cleanup-action to obtain branch |  | ||||||
|     runs-on: ubuntu-20.04 |  | ||||||
|     steps: |  | ||||||
|     - name: Cleanup previous runs on this branch |  | ||||||
|       uses: rokroskar/workflow-run-cleanup-action@v0.3.3 |  | ||||||
|       if: "!startsWith(github.ref, 'refs/tags/') && github.ref != 'refs/heads/stable' && github.repository == 'freqtrade/freqtrade'" |  | ||||||
|       env: |  | ||||||
|         GITHUB_TOKEN: "${{ secrets.GITHUB_TOKEN }}" |  | ||||||
|  |  | ||||||
|   # Notify only once - when CI completes (and after deploy) in case it's successfull |   # Notify only once - when CI completes (and after deploy) in case it's successfull | ||||||
|   notify-complete: |   notify-complete: | ||||||
|     needs: [ build_linux, build_macos, build_windows, docs_check, mypy_version_check ] |     needs: [ build_linux, build_macos, build_windows, docs_check, mypy_version_check ] | ||||||
| @@ -344,7 +336,7 @@ jobs: | |||||||
|     - uses: actions/checkout@v3 |     - uses: actions/checkout@v3 | ||||||
|  |  | ||||||
|     - name: Set up Python |     - name: Set up Python | ||||||
|       uses: actions/setup-python@v3 |       uses: actions/setup-python@v4 | ||||||
|       with: |       with: | ||||||
|         python-version: "3.9" |         python-version: "3.9" | ||||||
|  |  | ||||||
|   | |||||||
| @@ -14,8 +14,8 @@ repos: | |||||||
|         exclude: build_helpers |         exclude: build_helpers | ||||||
|         additional_dependencies: |         additional_dependencies: | ||||||
|           - types-cachetools==5.0.1 |           - types-cachetools==5.0.1 | ||||||
|           - types-filelock==3.2.6 |           - types-filelock==3.2.7 | ||||||
|           - types-requests==2.27.29 |           - types-requests==2.27.30 | ||||||
|           - types-tabulate==0.8.9 |           - types-tabulate==0.8.9 | ||||||
|           - types-python-dateutil==2.8.17 |           - types-python-dateutil==2.8.17 | ||||||
|         # stages: [push] |         # stages: [push] | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| FROM python:3.10.4-slim-bullseye as base | FROM python:3.10.5-slim-bullseye as base | ||||||
|  |  | ||||||
| # Setup env | # Setup env | ||||||
| ENV LANG C.UTF-8 | ENV LANG C.UTF-8 | ||||||
|   | |||||||
| @@ -22,50 +22,79 @@ DataFrame of the candles that resulted in buy signals. Depending on how many buy | |||||||
| makes, this file may get quite large, so periodically check your `user_data/backtest_results` | makes, this file may get quite large, so periodically check your `user_data/backtest_results` | ||||||
| folder to delete old exports. | folder to delete old exports. | ||||||
|  |  | ||||||
| To analyze the buy tags, we need to use the `buy_reasons.py` script from |  | ||||||
| [froggleston's repo](https://github.com/froggleston/freqtrade-buyreasons). Follow the instructions |  | ||||||
| in their README to copy the script into your `freqtrade/scripts/` folder. |  | ||||||
|  |  | ||||||
| Before running your next backtest, make sure you either delete your old backtest results or run | Before running your next backtest, make sure you either delete your old backtest results or run | ||||||
| backtesting with the `--cache none` option to make sure no cached results are used. | backtesting with the `--cache none` option to make sure no cached results are used. | ||||||
|  |  | ||||||
| If all goes well, you should now see a `backtest-result-{timestamp}_signals.pkl` file in the | If all goes well, you should now see a `backtest-result-{timestamp}_signals.pkl` file in the | ||||||
| `user_data/backtest_results` folder. | `user_data/backtest_results` folder. | ||||||
|  |  | ||||||
| Now run the `buy_reasons.py` script, supplying a few options: | To analyze the entry/exit tags, we now need to use the `freqtrade backtesting-analysis` command | ||||||
|  | with `--analysis-groups` option provided with space-separated arguments (default `0 1 2`): | ||||||
|  |  | ||||||
| ``` bash | ``` bash | ||||||
| python3 scripts/buy_reasons.py -c <config.json> -s <strategy_name> -t <timerange> -g0,1,2,3,4 | freqtrade backtesting-analysis -c <config.json> --analysis-groups 0 1 2 3 4 | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| The `-g` option is used to specify the various tabular outputs, ranging from the simplest (0) | This command will read from the last backtesting results. The `--analysis-groups` option is | ||||||
| to the most detailed per pair, per buy and per sell tag (4). More options are available by | used to specify the various tabular outputs showing the profit fo each group or trade, | ||||||
| running with the `-h` option. | ranging from the simplest (0) to the most detailed per pair, per buy and per sell tag (4): | ||||||
|  |  | ||||||
|  | * 1: profit summaries grouped by enter_tag | ||||||
|  | * 2: profit summaries grouped by enter_tag and exit_tag | ||||||
|  | * 3: profit summaries grouped by pair and enter_tag | ||||||
|  | * 4: profit summaries grouped by pair, enter_ and exit_tag (this can get quite large) | ||||||
|  |  | ||||||
|  | More options are available by running with the `-h` option. | ||||||
|  |  | ||||||
|  | ### Using export-filename | ||||||
|  |  | ||||||
|  | Normally, `backtesting-analysis` uses the latest backtest results, but if you wanted to go | ||||||
|  | back to a previous backtest output, you need to supply the `--export-filename` option. | ||||||
|  | You can supply the same parameter to `backtest-analysis` with the name of the final backtest | ||||||
|  | output file. This allows you to keep historical versions of backtest results and re-analyse | ||||||
|  | them at a later date: | ||||||
|  |  | ||||||
|  | ``` bash | ||||||
|  | freqtrade backtesting -c <config.json> --timeframe <tf> --strategy <strategy_name> --timerange=<timerange> --export=signals --export-filename=/tmp/mystrat_backtest.json | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | You should see some output similar to below in the logs with the name of the timestamped | ||||||
|  | filename that was exported: | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | 2022-06-14 16:28:32,698 - freqtrade.misc - INFO - dumping json to "/tmp/mystrat_backtest-2022-06-14_16-28-32.json" | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | You can then use that filename in `backtesting-analysis`: | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | freqtrade backtesting-analysis -c <config.json> --export-filename=/tmp/mystrat_backtest-2022-06-14_16-28-32.json | ||||||
|  | ``` | ||||||
|  |  | ||||||
| ### Tuning the buy tags and sell tags to display | ### Tuning the buy tags and sell tags to display | ||||||
|  |  | ||||||
| To show only certain buy and sell tags in the displayed output, use the following two options: | To show only certain buy and sell tags in the displayed output, use the following two options: | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| --enter_reason_list : Comma separated list of enter signals to analyse. Default: "all" | --enter-reason-list : Space-separated list of enter signals to analyse. Default: "all" | ||||||
| --exit_reason_list : Comma separated list of exit signals to analyse. Default: "stop_loss,trailing_stop_loss" | --exit-reason-list : Space-separated list of exit signals to analyse. Default: "all" | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| For example: | For example: | ||||||
|  |  | ||||||
| ```bash | ```bash | ||||||
| python3 scripts/buy_reasons.py -c <config.json> -s <strategy_name> -t <timerange> -g0,1,2,3,4 --enter_reason_list "enter_tag_a,enter_tag_b" --exit_reason_list "roi,custom_exit_tag_a,stop_loss" | freqtrade backtesting-analysis -c <config.json> --analysis-groups 0 2 --enter-reason-list enter_tag_a enter_tag_b --exit-reason-list roi custom_exit_tag_a stop_loss | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### Outputting signal candle indicators | ### Outputting signal candle indicators | ||||||
|  |  | ||||||
| The real power of the buy_reasons.py script comes from the ability to print out the indicator | The real power of `freqtrade backtesting-analysis` comes from the ability to print out the indicator | ||||||
| values present on signal candles to allow fine-grained investigation and tuning of buy signal | values present on signal candles to allow fine-grained investigation and tuning of buy signal | ||||||
| indicators. To print out a column for a given set of indicators, use the `--indicator-list` | indicators. To print out a column for a given set of indicators, use the `--indicator-list` | ||||||
| option: | option: | ||||||
|  |  | ||||||
| ```bash | ```bash | ||||||
| python3 scripts/buy_reasons.py -c <config.json> -s <strategy_name> -t <timerange> -g0,1,2,3,4 --enter_reason_list "enter_tag_a,enter_tag_b" --exit_reason_list "roi,custom_exit_tag_a,stop_loss" --indicator_list "rsi,rsi_1h,bb_lowerband,ema_9,macd,macdsignal" | freqtrade backtesting-analysis -c <config.json> --analysis-groups 0 2 --enter-reason-list enter_tag_a enter_tag_b --exit-reason-list roi custom_exit_tag_a stop_loss --indicator-list rsi rsi_1h bb_lowerband ema_9 macd macdsignal | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| The indicators have to be present in your strategy's main DataFrame (either for your main | The indicators have to be present in your strategy's main DataFrame (either for your main | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								docs/assets/discord_notification.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								docs/assets/discord_notification.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 48 KiB | 
| @@ -1,5 +1,5 @@ | |||||||
| mkdocs==1.3.0 | mkdocs==1.3.0 | ||||||
| mkdocs-material==8.2.16 | mkdocs-material==8.3.4 | ||||||
| mdx_truly_sane_lists==1.2 | mdx_truly_sane_lists==1.2 | ||||||
| pymdown-extensions==9.4 | pymdown-extensions==9.5 | ||||||
| jinja2==3.1.2 | jinja2==3.1.2 | ||||||
|   | |||||||
| @@ -89,11 +89,12 @@ WHERE id=31; | |||||||
|  |  | ||||||
| If you'd still like to remove a trade from the database directly, you can use the below query. | If you'd still like to remove a trade from the database directly, you can use the below query. | ||||||
|  |  | ||||||
| ```sql | !!! Danger | ||||||
| DELETE FROM trades WHERE id = <tradeid>; |     Some systems (Ubuntu) disable foreign keys in their sqlite3 packaging. When using sqlite - please ensure that foreign keys are on by running `PRAGMA foreign_keys = ON` before the above query. | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ```sql | ```sql | ||||||
|  | DELETE FROM trades WHERE id = <tradeid>; | ||||||
|  |  | ||||||
| DELETE FROM trades WHERE id = 31; | DELETE FROM trades WHERE id = 31; | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| @@ -102,13 +103,20 @@ DELETE FROM trades WHERE id = 31; | |||||||
|  |  | ||||||
| ## Use a different database system | ## Use a different database system | ||||||
|  |  | ||||||
|  | Freqtrade is using SQLAlchemy, which supports multiple different database systems. As such, a multitude of database systems should be supported. | ||||||
|  | Freqtrade does not depend or install any additional database driver. Please refer to the [SQLAlchemy docs](https://docs.sqlalchemy.org/en/14/core/engines.html#database-urls) on installation instructions for the respective database systems. | ||||||
|  |  | ||||||
|  | The following systems have been tested and are known to work with freqtrade: | ||||||
|  |  | ||||||
|  | * sqlite (default) | ||||||
|  | * PostgreSQL) | ||||||
|  | * MariaDB | ||||||
|  |  | ||||||
| !!! Warning | !!! Warning | ||||||
|     By using one of the below database systems, you acknowledge that you know how to manage such a system. Freqtrade will not provide any support with setup or maintenance (or backups) of the below database systems. |     By using one of the below database systems, you acknowledge that you know how to manage such a system. The freqtrade team will not provide any support with setup or maintenance (or backups) of the below database systems. | ||||||
|  |  | ||||||
| ### PostgreSQL | ### PostgreSQL | ||||||
|  |  | ||||||
| Freqtrade supports PostgreSQL by using SQLAlchemy, which supports multiple different database systems. |  | ||||||
|  |  | ||||||
| Installation: | Installation: | ||||||
| `pip install psycopg2-binary` | `pip install psycopg2-binary` | ||||||
|  |  | ||||||
|   | |||||||
| @@ -550,7 +550,8 @@ class AwesomeStrategy(IStrategy): | |||||||
|         :param pair: Pair that's about to be bought/shorted. |         :param pair: Pair that's about to be bought/shorted. | ||||||
|         :param order_type: Order type (as configured in order_types). usually limit or market. |         :param order_type: Order type (as configured in order_types). usually limit or market. | ||||||
|         :param amount: Amount in target (base) currency that's going to be traded. |         :param amount: Amount in target (base) currency that's going to be traded. | ||||||
|         :param rate: Rate that's going to be used when using limit orders |         :param rate: Rate that's going to be used when using limit orders  | ||||||
|  |                      or current rate for market orders. | ||||||
|         :param time_in_force: Time in force. Defaults to GTC (Good-til-cancelled). |         :param time_in_force: Time in force. Defaults to GTC (Good-til-cancelled). | ||||||
|         :param current_time: datetime object, containing the current datetime |         :param current_time: datetime object, containing the current datetime | ||||||
|         :param entry_tag: Optional entry_tag (buy_tag) if provided with the buy signal. |         :param entry_tag: Optional entry_tag (buy_tag) if provided with the buy signal. | ||||||
| @@ -600,6 +601,7 @@ class AwesomeStrategy(IStrategy): | |||||||
|         :param order_type: Order type (as configured in order_types). usually limit or market. |         :param order_type: Order type (as configured in order_types). usually limit or market. | ||||||
|         :param amount: Amount in base currency. |         :param amount: Amount in base currency. | ||||||
|         :param rate: Rate that's going to be used when using limit orders |         :param rate: Rate that's going to be used when using limit orders | ||||||
|  |                      or current rate for market orders. | ||||||
|         :param time_in_force: Time in force. Defaults to GTC (Good-til-cancelled). |         :param time_in_force: Time in force. Defaults to GTC (Good-til-cancelled). | ||||||
|         :param exit_reason: Exit reason. |         :param exit_reason: Exit reason. | ||||||
|             Can be any of ['roi', 'stop_loss', 'stoploss_on_exchange', 'trailing_stop_loss', |             Can be any of ['roi', 'stop_loss', 'stoploss_on_exchange', 'trailing_stop_loss', | ||||||
| @@ -804,17 +806,18 @@ For markets / exchanges that don't support leverage, this method is ignored. | |||||||
|  |  | ||||||
| ``` python | ``` python | ||||||
| class AwesomeStrategy(IStrategy): | class AwesomeStrategy(IStrategy): | ||||||
|     def leverage(self, pair: str, current_time: 'datetime', current_rate: float, |     def leverage(self, pair: str, current_time: datetime, current_rate: float, | ||||||
|                  proposed_leverage: float, max_leverage: float, side: str, |                  proposed_leverage: float, max_leverage: float, entry_tag: Optional[str], side: str, | ||||||
|                  **kwargs) -> float: |                  **kwargs) -> float: | ||||||
|         """ |         """ | ||||||
|         Customize leverage for each new trade. |         Customize leverage for each new trade. This method is only called in futures mode. | ||||||
|  |  | ||||||
|         :param pair: Pair that's currently analyzed |         :param pair: Pair that's currently analyzed | ||||||
|         :param current_time: datetime object, containing the current datetime |         :param current_time: datetime object, containing the current datetime | ||||||
|         :param current_rate: Rate, calculated based on pricing settings in exit_pricing. |         :param current_rate: Rate, calculated based on pricing settings in exit_pricing. | ||||||
|         :param proposed_leverage: A leverage proposed by the bot. |         :param proposed_leverage: A leverage proposed by the bot. | ||||||
|         :param max_leverage: Max leverage allowed on this pair |         :param max_leverage: Max leverage allowed on this pair | ||||||
|  |         :param entry_tag: Optional entry_tag (buy_tag) if provided with the buy signal. | ||||||
|         :param side: 'long' or 'short' - indicating the direction of the proposed trade |         :param side: 'long' or 'short' - indicating the direction of the proposed trade | ||||||
|         :return: A leverage amount, which is between 1.0 and max_leverage. |         :return: A leverage amount, which is between 1.0 and max_leverage. | ||||||
|         """ |         """ | ||||||
|   | |||||||
| @@ -328,11 +328,11 @@ Per default `/daily` will return the 7 last days. The example below if for `/dai | |||||||
|  |  | ||||||
| > **Daily Profit over the last 3 days:** | > **Daily Profit over the last 3 days:** | ||||||
| ``` | ``` | ||||||
| Day         Profit BTC      Profit USD | Day (count)     USDT          USD         Profit % | ||||||
| ----------  --------------  ------------ | --------------  ------------  ----------  ---------- | ||||||
| 2018-01-03  0.00224175 BTC  29,142 USD | 2022-06-11 (1)  -0.746 USDT   -0.75 USD   -0.08% | ||||||
| 2018-01-02  0.00033131 BTC   4,307 USD | 2022-06-10 (0)  0 USDT        0.00 USD    0.00% | ||||||
| 2018-01-01  0.00269130 BTC  34.986 USD | 2022-06-09 (5)  20 USDT       20.10 USD   5.00% | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### /weekly <n> | ### /weekly <n> | ||||||
| @@ -342,11 +342,11 @@ from Monday. The example below if for `/weekly 3`: | |||||||
|  |  | ||||||
| > **Weekly Profit over the last 3 weeks (starting from Monday):** | > **Weekly Profit over the last 3 weeks (starting from Monday):** | ||||||
| ``` | ``` | ||||||
| Monday         Profit BTC      Profit USD | Monday (count)  Profit BTC      Profit USD   Profit % | ||||||
| ----------  --------------  ------------ | -------------  --------------  ------------    ---------- | ||||||
| 2018-01-03  0.00224175 BTC  29,142 USD | 2018-01-03 (5)  0.00224175 BTC  29,142 USD   4.98% | ||||||
| 2017-12-27  0.00033131 BTC   4,307 USD | 2017-12-27 (1)  0.00033131 BTC   4,307 USD   0.00% | ||||||
| 2017-12-20  0.00269130 BTC  34.986 USD | 2017-12-20 (4)  0.00269130 BTC  34.986 USD   5.12% | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### /monthly <n> | ### /monthly <n> | ||||||
| @@ -356,11 +356,11 @@ if for `/monthly 3`: | |||||||
|  |  | ||||||
| > **Monthly Profit over the last 3 months:** | > **Monthly Profit over the last 3 months:** | ||||||
| ``` | ``` | ||||||
| Month         Profit BTC      Profit USD | Month (count)  Profit BTC      Profit USD    Profit % | ||||||
| ----------  --------------  ------------ | -------------  --------------  ------------    ---------- | ||||||
| 2018-01     0.00224175 BTC  29,142 USD | 2018-01 (20)    0.00224175 BTC  29,142 USD  4.98% | ||||||
| 2017-12     0.00033131 BTC   4,307 USD | 2017-12 (5)    0.00033131 BTC   4,307 USD   0.00% | ||||||
| 2017-11     0.00269130 BTC  34.986 USD | 2017-11 (10)    0.00269130 BTC  34.986 USD  5.10% | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### /whitelist | ### /whitelist | ||||||
|   | |||||||
| @@ -32,4 +32,8 @@ Please ensure that you're also updating dependencies - otherwise things might br | |||||||
| ``` bash | ``` bash | ||||||
| git pull | git pull | ||||||
| pip install -U -r requirements.txt | pip install -U -r requirements.txt | ||||||
|  | pip install -e . | ||||||
|  |  | ||||||
|  | # Ensure freqUI is at the latest version | ||||||
|  | freqtrade install-ui  | ||||||
| ``` | ``` | ||||||
|   | |||||||
| @@ -651,6 +651,61 @@ Common arguments: | |||||||
|  |  | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
|  | ## Detailed backtest analysis | ||||||
|  |  | ||||||
|  | Advanced backtest result analysis. | ||||||
|  |  | ||||||
|  | More details in the [Backtesting analysis](advanced-backtesting.md#analyze-the-buyentry-and-sellexit-tags) Section. | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | usage: freqtrade backtesting-analysis [-h] [-v] [--logfile FILE] [-V] | ||||||
|  |                                       [-c PATH] [-d PATH] [--userdir PATH] | ||||||
|  |                                       [--export-filename PATH] | ||||||
|  |                                       [--analysis-groups {0,1,2,3,4} [{0,1,2,3,4} ...]] | ||||||
|  |                                       [--enter-reason-list ENTER_REASON_LIST [ENTER_REASON_LIST ...]] | ||||||
|  |                                       [--exit-reason-list EXIT_REASON_LIST [EXIT_REASON_LIST ...]] | ||||||
|  |                                       [--indicator-list INDICATOR_LIST [INDICATOR_LIST ...]] | ||||||
|  |  | ||||||
|  | optional arguments: | ||||||
|  |   -h, --help            show this help message and exit | ||||||
|  |   --export-filename PATH, --backtest-filename PATH | ||||||
|  |                         Use this filename for backtest results.Requires | ||||||
|  |                         `--export` to be set as well. Example: `--export-filen | ||||||
|  |                         ame=user_data/backtest_results/backtest_today.json` | ||||||
|  |   --analysis-groups {0,1,2,3,4} [{0,1,2,3,4} ...] | ||||||
|  |                         grouping output - 0: simple wins/losses by enter tag, | ||||||
|  |                         1: by enter_tag, 2: by enter_tag and exit_tag, 3: by | ||||||
|  |                         pair and enter_tag, 4: by pair, enter_ and exit_tag | ||||||
|  |                         (this can get quite large) | ||||||
|  |   --enter-reason-list ENTER_REASON_LIST [ENTER_REASON_LIST ...] | ||||||
|  |                         Comma separated list of entry signals to analyse. | ||||||
|  |                         Default: all. e.g. 'entry_tag_a,entry_tag_b' | ||||||
|  |   --exit-reason-list EXIT_REASON_LIST [EXIT_REASON_LIST ...] | ||||||
|  |                         Comma separated list of exit signals to analyse. | ||||||
|  |                         Default: all. e.g. | ||||||
|  |                         'exit_tag_a,roi,stop_loss,trailing_stop_loss' | ||||||
|  |   --indicator-list INDICATOR_LIST [INDICATOR_LIST ...] | ||||||
|  |                         Comma separated list of indicators to analyse. e.g. | ||||||
|  |                         'close,rsi,bb_lowerband,profit_abs' | ||||||
|  |  | ||||||
|  | Common arguments: | ||||||
|  |   -v, --verbose         Verbose mode (-vv for more, -vvv to get all messages). | ||||||
|  |   --logfile FILE        Log to the file specified. Special values are: | ||||||
|  |                         'syslog', 'journald'. See the documentation for more | ||||||
|  |                         details. | ||||||
|  |   -V, --version         show program's version number and exit | ||||||
|  |   -c PATH, --config PATH | ||||||
|  |                         Specify configuration file (default: | ||||||
|  |                         `userdir/config.json` or `config.json` whichever | ||||||
|  |                         exists). Multiple --config options may be used. Can be | ||||||
|  |                         set to `-` to read config from stdin. | ||||||
|  |   -d PATH, --datadir PATH | ||||||
|  |                         Path to directory with historical backtesting data. | ||||||
|  |   --userdir PATH, --user-data-dir PATH | ||||||
|  |                         Path to userdata directory. | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  |  | ||||||
| ## List Hyperopt results | ## List Hyperopt results | ||||||
|  |  | ||||||
| You can list the hyperoptimization epochs the Hyperopt module evaluated previously with the `hyperopt-list` sub-command. | You can list the hyperoptimization epochs the Hyperopt module evaluated previously with the `hyperopt-list` sub-command. | ||||||
|   | |||||||
| @@ -239,3 +239,52 @@ Possible parameters are: | |||||||
| The fields in `webhook.webhookstatus` are used for regular status messages (Started / Stopped / ...). Parameters are filled using string.format. | The fields in `webhook.webhookstatus` are used for regular status messages (Started / Stopped / ...). Parameters are filled using string.format. | ||||||
|  |  | ||||||
| The only possible value here is `{status}`. | The only possible value here is `{status}`. | ||||||
|  |  | ||||||
|  | ## Discord | ||||||
|  |  | ||||||
|  | A special form of webhooks is available for discord. | ||||||
|  | You can configure this as follows: | ||||||
|  |  | ||||||
|  | ```json | ||||||
|  | "discord": { | ||||||
|  |     "enabled": true, | ||||||
|  |     "webhook_url": "https://discord.com/api/webhooks/<Your webhook URL ...>", | ||||||
|  |     "exit_fill": [ | ||||||
|  |         {"Trade ID": "{trade_id}"}, | ||||||
|  |         {"Exchange": "{exchange}"}, | ||||||
|  |         {"Pair": "{pair}"}, | ||||||
|  |         {"Direction": "{direction}"}, | ||||||
|  |         {"Open rate": "{open_rate}"}, | ||||||
|  |         {"Close rate": "{close_rate}"}, | ||||||
|  |         {"Amount": "{amount}"}, | ||||||
|  |         {"Open date": "{open_date:%Y-%m-%d %H:%M:%S}"}, | ||||||
|  |         {"Close date": "{close_date:%Y-%m-%d %H:%M:%S}"}, | ||||||
|  |         {"Profit": "{profit_amount} {stake_currency}"}, | ||||||
|  |         {"Profitability": "{profit_ratio:.2%}"}, | ||||||
|  |         {"Enter tag": "{enter_tag}"}, | ||||||
|  |         {"Exit Reason": "{exit_reason}"}, | ||||||
|  |         {"Strategy": "{strategy}"}, | ||||||
|  |         {"Timeframe": "{timeframe}"}, | ||||||
|  |     ], | ||||||
|  |     "entry_fill": [ | ||||||
|  |         {"Trade ID": "{trade_id}"}, | ||||||
|  |         {"Exchange": "{exchange}"}, | ||||||
|  |         {"Pair": "{pair}"}, | ||||||
|  |         {"Direction": "{direction}"}, | ||||||
|  |         {"Open rate": "{open_rate}"}, | ||||||
|  |         {"Amount": "{amount}"}, | ||||||
|  |         {"Open date": "{open_date:%Y-%m-%d %H:%M:%S}"}, | ||||||
|  |         {"Enter tag": "{enter_tag}"}, | ||||||
|  |         {"Strategy": "{strategy} {timeframe}"}, | ||||||
|  |     ] | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  |  | ||||||
|  | The above represents the default (`exit_fill` and `entry_fill` are optional and will default to the above configuration) - modifications are obviously possible. | ||||||
|  |  | ||||||
|  | Available fields correspond to the fields for webhooks and are documented in the corresponding webhook sections. | ||||||
|  |  | ||||||
|  | The notifications will look as follows by default. | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ Contains all start-commands, subcommands and CLI Interface creation. | |||||||
| Note: Be careful with file-scoped imports in these subfiles. | Note: Be careful with file-scoped imports in these subfiles. | ||||||
|     as they are parsed on startup, nothing containing optional modules should be loaded. |     as they are parsed on startup, nothing containing optional modules should be loaded. | ||||||
| """ | """ | ||||||
|  | from freqtrade.commands.analyze_commands import start_analysis_entries_exits | ||||||
| from freqtrade.commands.arguments import Arguments | from freqtrade.commands.arguments import Arguments | ||||||
| from freqtrade.commands.build_config_commands import start_new_config | from freqtrade.commands.build_config_commands import start_new_config | ||||||
| from freqtrade.commands.data_commands import (start_convert_data, start_convert_trades, | from freqtrade.commands.data_commands import (start_convert_data, start_convert_trades, | ||||||
|   | |||||||
							
								
								
									
										69
									
								
								freqtrade/commands/analyze_commands.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										69
									
								
								freqtrade/commands/analyze_commands.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,69 @@ | |||||||
|  | import logging | ||||||
|  | from pathlib import Path | ||||||
|  | from typing import Any, Dict | ||||||
|  |  | ||||||
|  | from freqtrade.configuration import setup_utils_configuration | ||||||
|  | from freqtrade.enums import RunMode | ||||||
|  | from freqtrade.exceptions import OperationalException | ||||||
|  |  | ||||||
|  |  | ||||||
|  | logger = logging.getLogger(__name__) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def setup_analyze_configuration(args: Dict[str, Any], method: RunMode) -> Dict[str, Any]: | ||||||
|  |     """ | ||||||
|  |     Prepare the configuration for the entry/exit reason analysis module | ||||||
|  |     :param args: Cli args from Arguments() | ||||||
|  |     :param method: Bot running mode | ||||||
|  |     :return: Configuration | ||||||
|  |     """ | ||||||
|  |     config = setup_utils_configuration(args, method) | ||||||
|  |  | ||||||
|  |     no_unlimited_runmodes = { | ||||||
|  |         RunMode.BACKTEST: 'backtesting', | ||||||
|  |     } | ||||||
|  |     if method in no_unlimited_runmodes.keys(): | ||||||
|  |         from freqtrade.data.btanalysis import get_latest_backtest_filename | ||||||
|  |  | ||||||
|  |         if 'exportfilename' in config: | ||||||
|  |             if config['exportfilename'].is_dir(): | ||||||
|  |                 btfile = Path(get_latest_backtest_filename(config['exportfilename'])) | ||||||
|  |                 signals_file = f"{config['exportfilename']}/{btfile.stem}_signals.pkl" | ||||||
|  |             else: | ||||||
|  |                 if config['exportfilename'].exists(): | ||||||
|  |                     btfile = Path(config['exportfilename']) | ||||||
|  |                     signals_file = f"{btfile.parent}/{btfile.stem}_signals.pkl" | ||||||
|  |                 else: | ||||||
|  |                     raise OperationalException(f"{config['exportfilename']} does not exist.") | ||||||
|  |         else: | ||||||
|  |             raise OperationalException('exportfilename not in config.') | ||||||
|  |  | ||||||
|  |         if (not Path(signals_file).exists()): | ||||||
|  |             raise OperationalException( | ||||||
|  |                 (f"Cannot find latest backtest signals file: {signals_file}." | ||||||
|  |                   "Run backtesting with `--export signals`.") | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |     return config | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def start_analysis_entries_exits(args: Dict[str, Any]) -> None: | ||||||
|  |     """ | ||||||
|  |     Start analysis script | ||||||
|  |     :param args: Cli args from Arguments() | ||||||
|  |     :return: None | ||||||
|  |     """ | ||||||
|  |     from freqtrade.data.entryexitanalysis import process_entry_exit_reasons | ||||||
|  |  | ||||||
|  |     # Initialize configuration | ||||||
|  |     config = setup_analyze_configuration(args, RunMode.BACKTEST) | ||||||
|  |  | ||||||
|  |     logger.info('Starting freqtrade in analysis mode') | ||||||
|  |  | ||||||
|  |     process_entry_exit_reasons(config['exportfilename'], | ||||||
|  |                                config['exchange']['pair_whitelist'], | ||||||
|  |                                config['analysis_groups'], | ||||||
|  |                                config['enter_reason_list'], | ||||||
|  |                                config['exit_reason_list'], | ||||||
|  |                                config['indicator_list'] | ||||||
|  |                                ) | ||||||
| @@ -101,6 +101,9 @@ ARGS_HYPEROPT_SHOW = ["hyperopt_list_best", "hyperopt_list_profitable", "hyperop | |||||||
|                       "print_json", "hyperoptexportfilename", "hyperopt_show_no_header", |                       "print_json", "hyperoptexportfilename", "hyperopt_show_no_header", | ||||||
|                       "disableparamexport", "backtest_breakdown"] |                       "disableparamexport", "backtest_breakdown"] | ||||||
|  |  | ||||||
|  | ARGS_ANALYZE_ENTRIES_EXITS = ["exportfilename", "analysis_groups", "enter_reason_list", | ||||||
|  |                               "exit_reason_list", "indicator_list"] | ||||||
|  |  | ||||||
| NO_CONF_REQURIED = ["convert-data", "convert-trade-data", "download-data", "list-timeframes", | NO_CONF_REQURIED = ["convert-data", "convert-trade-data", "download-data", "list-timeframes", | ||||||
|                     "list-markets", "list-pairs", "list-strategies", "list-data", |                     "list-markets", "list-pairs", "list-strategies", "list-data", | ||||||
|                     "hyperopt-list", "hyperopt-show", "backtest-filter", |                     "hyperopt-list", "hyperopt-show", "backtest-filter", | ||||||
| @@ -182,8 +185,9 @@ class Arguments: | |||||||
|         self.parser = argparse.ArgumentParser(description='Free, open source crypto trading bot') |         self.parser = argparse.ArgumentParser(description='Free, open source crypto trading bot') | ||||||
|         self._build_args(optionlist=['version'], parser=self.parser) |         self._build_args(optionlist=['version'], parser=self.parser) | ||||||
|  |  | ||||||
|         from freqtrade.commands import (start_backtesting, start_backtesting_show, |         from freqtrade.commands import (start_analysis_entries_exits, start_backtesting, | ||||||
|                                         start_convert_data, start_convert_db, start_convert_trades, |                                         start_backtesting_show, start_convert_data, | ||||||
|  |                                         start_convert_db, start_convert_trades, | ||||||
|                                         start_create_userdir, start_download_data, start_edge, |                                         start_create_userdir, start_download_data, start_edge, | ||||||
|                                         start_hyperopt, start_hyperopt_list, start_hyperopt_show, |                                         start_hyperopt, start_hyperopt_list, start_hyperopt_show, | ||||||
|                                         start_install_ui, start_list_data, start_list_exchanges, |                                         start_install_ui, start_list_data, start_list_exchanges, | ||||||
| @@ -283,6 +287,13 @@ class Arguments: | |||||||
|         backtesting_show_cmd.set_defaults(func=start_backtesting_show) |         backtesting_show_cmd.set_defaults(func=start_backtesting_show) | ||||||
|         self._build_args(optionlist=ARGS_BACKTEST_SHOW, parser=backtesting_show_cmd) |         self._build_args(optionlist=ARGS_BACKTEST_SHOW, parser=backtesting_show_cmd) | ||||||
|  |  | ||||||
|  |         # Add backtesting analysis subcommand | ||||||
|  |         analysis_cmd = subparsers.add_parser('backtesting-analysis', | ||||||
|  |                                              help='Backtest Analysis module.', | ||||||
|  |                                              parents=[_common_parser]) | ||||||
|  |         analysis_cmd.set_defaults(func=start_analysis_entries_exits) | ||||||
|  |         self._build_args(optionlist=ARGS_ANALYZE_ENTRIES_EXITS, parser=analysis_cmd) | ||||||
|  |  | ||||||
|         # Add edge subcommand |         # Add edge subcommand | ||||||
|         edge_cmd = subparsers.add_parser('edge', help='Edge module.', |         edge_cmd = subparsers.add_parser('edge', help='Edge module.', | ||||||
|                                          parents=[_common_parser, _strategy_parser]) |                                          parents=[_common_parser, _strategy_parser]) | ||||||
|   | |||||||
| @@ -614,4 +614,37 @@ AVAILABLE_CLI_OPTIONS = { | |||||||
|               "that do not contain any parameters."), |               "that do not contain any parameters."), | ||||||
|         action="store_true", |         action="store_true", | ||||||
|     ), |     ), | ||||||
|  |     "analysis_groups": Arg( | ||||||
|  |         "--analysis-groups", | ||||||
|  |         help=("grouping output - " | ||||||
|  |               "0: simple wins/losses by enter tag, " | ||||||
|  |               "1: by enter_tag, " | ||||||
|  |               "2: by enter_tag and exit_tag, " | ||||||
|  |               "3: by pair and enter_tag, " | ||||||
|  |               "4: by pair, enter_ and exit_tag (this can get quite large)"), | ||||||
|  |         nargs='+', | ||||||
|  |         default=['0', '1', '2'], | ||||||
|  |         choices=['0', '1', '2', '3', '4'], | ||||||
|  |     ), | ||||||
|  |     "enter_reason_list": Arg( | ||||||
|  |         "--enter-reason-list", | ||||||
|  |         help=("Comma separated list of entry signals to analyse. Default: all. " | ||||||
|  |               "e.g. 'entry_tag_a,entry_tag_b'"), | ||||||
|  |         nargs='+', | ||||||
|  |         default=['all'], | ||||||
|  |     ), | ||||||
|  |     "exit_reason_list": Arg( | ||||||
|  |         "--exit-reason-list", | ||||||
|  |         help=("Comma separated list of exit signals to analyse. Default: all. " | ||||||
|  |               "e.g. 'exit_tag_a,roi,stop_loss,trailing_stop_loss'"), | ||||||
|  |         nargs='+', | ||||||
|  |         default=['all'], | ||||||
|  |     ), | ||||||
|  |     "indicator_list": Arg( | ||||||
|  |         "--indicator-list", | ||||||
|  |         help=("Comma separated list of indicators to analyse. " | ||||||
|  |               "e.g. 'close,rsi,bb_lowerband,profit_abs'"), | ||||||
|  |         nargs='+', | ||||||
|  |         default=[], | ||||||
|  |     ), | ||||||
| } | } | ||||||
|   | |||||||
| @@ -95,6 +95,8 @@ class Configuration: | |||||||
|  |  | ||||||
|         self._process_data_options(config) |         self._process_data_options(config) | ||||||
|  |  | ||||||
|  |         self._process_analyze_options(config) | ||||||
|  |  | ||||||
|         # Check if the exchange set by the user is supported |         # Check if the exchange set by the user is supported | ||||||
|         check_exchange(config, config.get('experimental', {}).get('block_bad_exchanges', True)) |         check_exchange(config, config.get('experimental', {}).get('block_bad_exchanges', True)) | ||||||
|  |  | ||||||
| @@ -433,6 +435,19 @@ class Configuration: | |||||||
|         self._args_to_config(config, argname='candle_types', |         self._args_to_config(config, argname='candle_types', | ||||||
|                              logstring='Detected --candle-types: {}') |                              logstring='Detected --candle-types: {}') | ||||||
|  |  | ||||||
|  |     def _process_analyze_options(self, config: Dict[str, Any]) -> None: | ||||||
|  |         self._args_to_config(config, argname='analysis_groups', | ||||||
|  |                              logstring='Analysis reason groups: {}') | ||||||
|  |  | ||||||
|  |         self._args_to_config(config, argname='enter_reason_list', | ||||||
|  |                              logstring='Analysis enter tag list: {}') | ||||||
|  |  | ||||||
|  |         self._args_to_config(config, argname='exit_reason_list', | ||||||
|  |                              logstring='Analysis exit tag list: {}') | ||||||
|  |  | ||||||
|  |         self._args_to_config(config, argname='indicator_list', | ||||||
|  |                              logstring='Analysis indicator list: {}') | ||||||
|  |  | ||||||
|     def _process_runmode(self, config: Dict[str, Any]) -> None: |     def _process_runmode(self, config: Dict[str, Any]) -> None: | ||||||
|  |  | ||||||
|         self._args_to_config(config, argname='dry_run', |         self._args_to_config(config, argname='dry_run', | ||||||
|   | |||||||
| @@ -336,6 +336,47 @@ CONF_SCHEMA = { | |||||||
|                 'webhookstatus': {'type': 'object'}, |                 'webhookstatus': {'type': 'object'}, | ||||||
|             }, |             }, | ||||||
|         }, |         }, | ||||||
|  |         'discord': { | ||||||
|  |             'type': 'object', | ||||||
|  |             'properties': { | ||||||
|  |                 'enabled': {'type': 'boolean'}, | ||||||
|  |                 'webhook_url': {'type': 'string'}, | ||||||
|  |                 "exit_fill": { | ||||||
|  |                     'type': 'array', 'items': {'type': 'object'}, | ||||||
|  |                     'default': [ | ||||||
|  |                         {"Trade ID": "{trade_id}"}, | ||||||
|  |                         {"Exchange": "{exchange}"}, | ||||||
|  |                         {"Pair": "{pair}"}, | ||||||
|  |                         {"Direction": "{direction}"}, | ||||||
|  |                         {"Open rate": "{open_rate}"}, | ||||||
|  |                         {"Close rate": "{close_rate}"}, | ||||||
|  |                         {"Amount": "{amount}"}, | ||||||
|  |                         {"Open date": "{open_date:%Y-%m-%d %H:%M:%S}"}, | ||||||
|  |                         {"Close date": "{close_date:%Y-%m-%d %H:%M:%S}"}, | ||||||
|  |                         {"Profit": "{profit_amount} {stake_currency}"}, | ||||||
|  |                         {"Profitability": "{profit_ratio:.2%}"}, | ||||||
|  |                         {"Enter tag": "{enter_tag}"}, | ||||||
|  |                         {"Exit Reason": "{exit_reason}"}, | ||||||
|  |                         {"Strategy": "{strategy}"}, | ||||||
|  |                         {"Timeframe": "{timeframe}"}, | ||||||
|  |                     ] | ||||||
|  |                 }, | ||||||
|  |                 "entry_fill": { | ||||||
|  |                     'type': 'array', 'items': {'type': 'object'}, | ||||||
|  |                     'default': [ | ||||||
|  |                         {"Trade ID": "{trade_id}"}, | ||||||
|  |                         {"Exchange": "{exchange}"}, | ||||||
|  |                         {"Pair": "{pair}"}, | ||||||
|  |                         {"Direction": "{direction}"}, | ||||||
|  |                         {"Open rate": "{open_rate}"}, | ||||||
|  |                         {"Amount": "{amount}"}, | ||||||
|  |                         {"Open date": "{open_date:%Y-%m-%d %H:%M:%S}"}, | ||||||
|  |                         {"Enter tag": "{enter_tag}"}, | ||||||
|  |                         {"Strategy": "{strategy} {timeframe}"}, | ||||||
|  |                     ] | ||||||
|  |                 }, | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|         'api_server': { |         'api_server': { | ||||||
|             'type': 'object', |             'type': 'object', | ||||||
|             'properties': { |             'properties': { | ||||||
|   | |||||||
| @@ -26,7 +26,7 @@ BT_DATA_COLUMNS = ['pair', 'stake_amount', 'amount', 'open_date', 'close_date', | |||||||
|                    'profit_ratio', 'profit_abs', 'exit_reason', |                    'profit_ratio', 'profit_abs', 'exit_reason', | ||||||
|                    'initial_stop_loss_abs', 'initial_stop_loss_ratio', 'stop_loss_abs', |                    'initial_stop_loss_abs', 'initial_stop_loss_ratio', 'stop_loss_abs', | ||||||
|                    'stop_loss_ratio', 'min_rate', 'max_rate', 'is_open', 'enter_tag', |                    'stop_loss_ratio', 'min_rate', 'max_rate', 'is_open', 'enter_tag', | ||||||
|                    'is_short' |                    'is_short', 'open_timestamp', 'close_timestamp', 'orders' | ||||||
|                    ] |                    ] | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -283,6 +283,8 @@ def load_backtest_data(filename: Union[Path, str], strategy: Optional[str] = Non | |||||||
|             if 'enter_tag' not in df.columns: |             if 'enter_tag' not in df.columns: | ||||||
|                 df['enter_tag'] = df['buy_tag'] |                 df['enter_tag'] = df['buy_tag'] | ||||||
|                 df = df.drop(['buy_tag'], axis=1) |                 df = df.drop(['buy_tag'], axis=1) | ||||||
|  |             if 'orders' not in df.columns: | ||||||
|  |                 df.loc[:, 'orders'] = None | ||||||
|  |  | ||||||
|     else: |     else: | ||||||
|         # old format - only with lists. |         # old format - only with lists. | ||||||
| @@ -337,7 +339,7 @@ def trade_list_to_dataframe(trades: List[LocalTrade]) -> pd.DataFrame: | |||||||
|     :param trades: List of trade objects |     :param trades: List of trade objects | ||||||
|     :return: Dataframe with BT_DATA_COLUMNS |     :return: Dataframe with BT_DATA_COLUMNS | ||||||
|     """ |     """ | ||||||
|     df = pd.DataFrame.from_records([t.to_json() for t in trades], columns=BT_DATA_COLUMNS) |     df = pd.DataFrame.from_records([t.to_json(True) for t in trades], columns=BT_DATA_COLUMNS) | ||||||
|     if len(df) > 0: |     if len(df) > 0: | ||||||
|         df.loc[:, 'close_date'] = pd.to_datetime(df['close_date'], utc=True) |         df.loc[:, 'close_date'] = pd.to_datetime(df['close_date'], utc=True) | ||||||
|         df.loc[:, 'open_date'] = pd.to_datetime(df['open_date'], utc=True) |         df.loc[:, 'open_date'] = pd.to_datetime(df['open_date'], utc=True) | ||||||
|   | |||||||
							
								
								
									
										227
									
								
								freqtrade/data/entryexitanalysis.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										227
									
								
								freqtrade/data/entryexitanalysis.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,227 @@ | |||||||
|  | import logging | ||||||
|  | from pathlib import Path | ||||||
|  | from typing import List, Optional | ||||||
|  |  | ||||||
|  | import joblib | ||||||
|  | import pandas as pd | ||||||
|  | from tabulate import tabulate | ||||||
|  |  | ||||||
|  | from freqtrade.data.btanalysis import (get_latest_backtest_filename, load_backtest_data, | ||||||
|  |                                        load_backtest_stats) | ||||||
|  | from freqtrade.exceptions import OperationalException | ||||||
|  |  | ||||||
|  |  | ||||||
|  | logger = logging.getLogger(__name__) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def _load_signal_candles(backtest_dir: Path): | ||||||
|  |     if backtest_dir.is_dir(): | ||||||
|  |         scpf = Path(backtest_dir, | ||||||
|  |                     Path(get_latest_backtest_filename(backtest_dir)).stem + "_signals.pkl" | ||||||
|  |                     ) | ||||||
|  |     else: | ||||||
|  |         scpf = Path(backtest_dir.parent / f"{backtest_dir.stem}_signals.pkl") | ||||||
|  |  | ||||||
|  |     try: | ||||||
|  |         scp = open(scpf, "rb") | ||||||
|  |         signal_candles = joblib.load(scp) | ||||||
|  |         logger.info(f"Loaded signal candles: {str(scpf)}") | ||||||
|  |     except Exception as e: | ||||||
|  |         logger.error("Cannot load signal candles from pickled results: ", e) | ||||||
|  |  | ||||||
|  |     return signal_candles | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def _process_candles_and_indicators(pairlist, strategy_name, trades, signal_candles): | ||||||
|  |     analysed_trades_dict = {} | ||||||
|  |     analysed_trades_dict[strategy_name] = {} | ||||||
|  |  | ||||||
|  |     try: | ||||||
|  |         logger.info(f"Processing {strategy_name} : {len(pairlist)} pairs") | ||||||
|  |  | ||||||
|  |         for pair in pairlist: | ||||||
|  |             if pair in signal_candles[strategy_name]: | ||||||
|  |                 analysed_trades_dict[strategy_name][pair] = _analyze_candles_and_indicators( | ||||||
|  |                                                               pair, | ||||||
|  |                                                               trades, | ||||||
|  |                                                               signal_candles[strategy_name][pair]) | ||||||
|  |     except Exception as e: | ||||||
|  |         print(f"Cannot process entry/exit reasons for {strategy_name}: ", e) | ||||||
|  |  | ||||||
|  |     return analysed_trades_dict | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def _analyze_candles_and_indicators(pair, trades, signal_candles): | ||||||
|  |     buyf = signal_candles | ||||||
|  |  | ||||||
|  |     if len(buyf) > 0: | ||||||
|  |         buyf = buyf.set_index('date', drop=False) | ||||||
|  |         trades_red = trades.loc[trades['pair'] == pair].copy() | ||||||
|  |  | ||||||
|  |         trades_inds = pd.DataFrame() | ||||||
|  |  | ||||||
|  |         if trades_red.shape[0] > 0 and buyf.shape[0] > 0: | ||||||
|  |             for t, v in trades_red.open_date.items(): | ||||||
|  |                 allinds = buyf.loc[(buyf['date'] < v)] | ||||||
|  |                 if allinds.shape[0] > 0: | ||||||
|  |                     tmp_inds = allinds.iloc[[-1]] | ||||||
|  |  | ||||||
|  |                     trades_red.loc[t, 'signal_date'] = tmp_inds['date'].values[0] | ||||||
|  |                     trades_red.loc[t, 'enter_reason'] = trades_red.loc[t, 'enter_tag'] | ||||||
|  |                     tmp_inds.index.rename('signal_date', inplace=True) | ||||||
|  |                     trades_inds = pd.concat([trades_inds, tmp_inds]) | ||||||
|  |  | ||||||
|  |             if 'signal_date' in trades_red: | ||||||
|  |                 trades_red['signal_date'] = pd.to_datetime(trades_red['signal_date'], utc=True) | ||||||
|  |                 trades_red.set_index('signal_date', inplace=True) | ||||||
|  |  | ||||||
|  |                 try: | ||||||
|  |                     trades_red = pd.merge(trades_red, trades_inds, on='signal_date', how='outer') | ||||||
|  |                 except Exception as e: | ||||||
|  |                     raise e | ||||||
|  |         return trades_red | ||||||
|  |     else: | ||||||
|  |         return pd.DataFrame() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def _do_group_table_output(bigdf, glist): | ||||||
|  |     for g in glist: | ||||||
|  |         # 0: summary wins/losses grouped by enter tag | ||||||
|  |         if g == "0": | ||||||
|  |             group_mask = ['enter_reason'] | ||||||
|  |             wins = bigdf.loc[bigdf['profit_abs'] >= 0] \ | ||||||
|  |                         .groupby(group_mask) \ | ||||||
|  |                         .agg({'profit_abs': ['sum']}) | ||||||
|  |  | ||||||
|  |             wins.columns = ['profit_abs_wins'] | ||||||
|  |             loss = bigdf.loc[bigdf['profit_abs'] < 0] \ | ||||||
|  |                         .groupby(group_mask) \ | ||||||
|  |                         .agg({'profit_abs': ['sum']}) | ||||||
|  |             loss.columns = ['profit_abs_loss'] | ||||||
|  |  | ||||||
|  |             new = bigdf.groupby(group_mask).agg({'profit_abs': [ | ||||||
|  |                                                     'count', | ||||||
|  |                                                     lambda x: sum(x > 0), | ||||||
|  |                                                     lambda x: sum(x <= 0)]}) | ||||||
|  |             new = pd.concat([new, wins, loss], axis=1).fillna(0) | ||||||
|  |  | ||||||
|  |             new['profit_tot'] = new['profit_abs_wins'] - abs(new['profit_abs_loss']) | ||||||
|  |             new['wl_ratio_pct'] = (new.iloc[:, 1] / new.iloc[:, 0] * 100).fillna(0) | ||||||
|  |             new['avg_win'] = (new['profit_abs_wins'] / new.iloc[:, 1]).fillna(0) | ||||||
|  |             new['avg_loss'] = (new['profit_abs_loss'] / new.iloc[:, 2]).fillna(0) | ||||||
|  |  | ||||||
|  |             new.columns = ['total_num_buys', 'wins', 'losses', 'profit_abs_wins', 'profit_abs_loss', | ||||||
|  |                            'profit_tot', 'wl_ratio_pct', 'avg_win', 'avg_loss'] | ||||||
|  |  | ||||||
|  |             sortcols = ['total_num_buys'] | ||||||
|  |  | ||||||
|  |             _print_table(new, sortcols, show_index=True) | ||||||
|  |  | ||||||
|  |         else: | ||||||
|  |             agg_mask = {'profit_abs': ['count', 'sum', 'median', 'mean'], | ||||||
|  |                         'profit_ratio': ['sum', 'median', 'mean']} | ||||||
|  |             agg_cols = ['num_buys', 'profit_abs_sum', 'profit_abs_median', | ||||||
|  |                         'profit_abs_mean', 'median_profit_pct', 'mean_profit_pct', | ||||||
|  |                         'total_profit_pct'] | ||||||
|  |             sortcols = ['profit_abs_sum', 'enter_reason'] | ||||||
|  |  | ||||||
|  |             # 1: profit summaries grouped by enter_tag | ||||||
|  |             if g == "1": | ||||||
|  |                 group_mask = ['enter_reason'] | ||||||
|  |  | ||||||
|  |             # 2: profit summaries grouped by enter_tag and exit_tag | ||||||
|  |             if g == "2": | ||||||
|  |                 group_mask = ['enter_reason', 'exit_reason'] | ||||||
|  |  | ||||||
|  |             # 3: profit summaries grouped by pair and enter_tag | ||||||
|  |             if g == "3": | ||||||
|  |                 group_mask = ['pair', 'enter_reason'] | ||||||
|  |  | ||||||
|  |             # 4: profit summaries grouped by pair, enter_ and exit_tag (this can get quite large) | ||||||
|  |             if g == "4": | ||||||
|  |                 group_mask = ['pair', 'enter_reason', 'exit_reason'] | ||||||
|  |             if group_mask: | ||||||
|  |                 new = bigdf.groupby(group_mask).agg(agg_mask).reset_index() | ||||||
|  |                 new.columns = group_mask + agg_cols | ||||||
|  |                 new['median_profit_pct'] = new['median_profit_pct'] * 100 | ||||||
|  |                 new['mean_profit_pct'] = new['mean_profit_pct'] * 100 | ||||||
|  |                 new['total_profit_pct'] = new['total_profit_pct'] * 100 | ||||||
|  |  | ||||||
|  |                 _print_table(new, sortcols) | ||||||
|  |             else: | ||||||
|  |                 logger.warning("Invalid group mask specified.") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def _print_results(analysed_trades, stratname, analysis_groups, | ||||||
|  |                    enter_reason_list, exit_reason_list, | ||||||
|  |                    indicator_list, columns=None): | ||||||
|  |     if columns is None: | ||||||
|  |         columns = ['pair', 'open_date', 'close_date', 'profit_abs', 'enter_reason', 'exit_reason'] | ||||||
|  |  | ||||||
|  |     bigdf = pd.DataFrame() | ||||||
|  |     for pair, trades in analysed_trades[stratname].items(): | ||||||
|  |         bigdf = pd.concat([bigdf, trades], ignore_index=True) | ||||||
|  |  | ||||||
|  |     if bigdf.shape[0] > 0 and ('enter_reason' in bigdf.columns): | ||||||
|  |         if analysis_groups: | ||||||
|  |             _do_group_table_output(bigdf, analysis_groups) | ||||||
|  |  | ||||||
|  |         if enter_reason_list and "all" not in enter_reason_list: | ||||||
|  |             bigdf = bigdf.loc[(bigdf['enter_reason'].isin(enter_reason_list))] | ||||||
|  |  | ||||||
|  |         if exit_reason_list and "all" not in exit_reason_list: | ||||||
|  |             bigdf = bigdf.loc[(bigdf['exit_reason'].isin(exit_reason_list))] | ||||||
|  |  | ||||||
|  |         if "all" in indicator_list: | ||||||
|  |             print(bigdf) | ||||||
|  |         elif indicator_list is not None: | ||||||
|  |             available_inds = [] | ||||||
|  |             for ind in indicator_list: | ||||||
|  |                 if ind in bigdf: | ||||||
|  |                     available_inds.append(ind) | ||||||
|  |             ilist = ["pair", "enter_reason", "exit_reason"] + available_inds | ||||||
|  |             _print_table(bigdf[ilist], sortcols=['exit_reason'], show_index=False) | ||||||
|  |     else: | ||||||
|  |         print("\\_ No trades to show") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def _print_table(df, sortcols=None, show_index=False): | ||||||
|  |     if (sortcols is not None): | ||||||
|  |         data = df.sort_values(sortcols) | ||||||
|  |     else: | ||||||
|  |         data = df | ||||||
|  |  | ||||||
|  |     print( | ||||||
|  |         tabulate( | ||||||
|  |             data, | ||||||
|  |             headers='keys', | ||||||
|  |             tablefmt='psql', | ||||||
|  |             showindex=show_index | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def process_entry_exit_reasons(backtest_dir: Path, | ||||||
|  |                                pairlist: List[str], | ||||||
|  |                                analysis_groups: Optional[List[str]] = ["0", "1", "2"], | ||||||
|  |                                enter_reason_list: Optional[List[str]] = ["all"], | ||||||
|  |                                exit_reason_list: Optional[List[str]] = ["all"], | ||||||
|  |                                indicator_list: Optional[List[str]] = []): | ||||||
|  |     try: | ||||||
|  |         backtest_stats = load_backtest_stats(backtest_dir) | ||||||
|  |         for strategy_name, results in backtest_stats['strategy'].items(): | ||||||
|  |             trades = load_backtest_data(backtest_dir, strategy_name) | ||||||
|  |  | ||||||
|  |             if not trades.empty: | ||||||
|  |                 signal_candles = _load_signal_candles(backtest_dir) | ||||||
|  |                 analysed_trades_dict = _process_candles_and_indicators(pairlist, strategy_name, | ||||||
|  |                                                                        trades, signal_candles) | ||||||
|  |                 _print_results(analysed_trades_dict, | ||||||
|  |                                strategy_name, | ||||||
|  |                                analysis_groups, | ||||||
|  |                                enter_reason_list, | ||||||
|  |                                exit_reason_list, | ||||||
|  |                                indicator_list) | ||||||
|  |  | ||||||
|  |     except ValueError as e: | ||||||
|  |         raise OperationalException(e) from e | ||||||
| @@ -4,7 +4,7 @@ Freqtrade is the main module of this bot. It contains the class Freqtrade() | |||||||
| import copy | import copy | ||||||
| import logging | import logging | ||||||
| import traceback | import traceback | ||||||
| from datetime import datetime, time, timezone | from datetime import datetime, time, timedelta, timezone | ||||||
| from math import isclose | from math import isclose | ||||||
| from threading import Lock | from threading import Lock | ||||||
| from typing import Any, Dict, List, Optional, Tuple | from typing import Any, Dict, List, Optional, Tuple | ||||||
| @@ -73,8 +73,6 @@ class FreqtradeBot(LoggingMixin): | |||||||
|  |  | ||||||
|         PairLocks.timeframe = self.config['timeframe'] |         PairLocks.timeframe = self.config['timeframe'] | ||||||
|  |  | ||||||
|         self.protections = ProtectionManager(self.config, self.strategy.protections) |  | ||||||
|  |  | ||||||
|         # RPC runs in separate threads, can start handling external commands just after |         # RPC runs in separate threads, can start handling external commands just after | ||||||
|         # initialization, even before Freqtradebot has a chance to start its throttling, |         # initialization, even before Freqtradebot has a chance to start its throttling, | ||||||
|         # so anything in the Freqtradebot instance should be ready (initialized), including |         # so anything in the Freqtradebot instance should be ready (initialized), including | ||||||
| @@ -124,6 +122,8 @@ class FreqtradeBot(LoggingMixin): | |||||||
|         self.last_process = datetime(1970, 1, 1, tzinfo=timezone.utc) |         self.last_process = datetime(1970, 1, 1, tzinfo=timezone.utc) | ||||||
|  |  | ||||||
|         self.strategy.ft_bot_start() |         self.strategy.ft_bot_start() | ||||||
|  |         # Initialize protections AFTER bot start - otherwise parameters are not loaded. | ||||||
|  |         self.protections = ProtectionManager(self.config, self.strategy.protections) | ||||||
|  |  | ||||||
|     def notify_status(self, msg: str) -> None: |     def notify_status(self, msg: str) -> None: | ||||||
|         """ |         """ | ||||||
| @@ -227,7 +227,7 @@ class FreqtradeBot(LoggingMixin): | |||||||
|         Notify the user when the bot is stopped (not reloaded) |         Notify the user when the bot is stopped (not reloaded) | ||||||
|         and there are still open trades active. |         and there are still open trades active. | ||||||
|         """ |         """ | ||||||
|         open_trades = Trade.get_trades([Trade.is_open.is_(True)]).all() |         open_trades = Trade.get_open_trades() | ||||||
|  |  | ||||||
|         if len(open_trades) != 0 and self.state != State.RELOAD_CONFIG: |         if len(open_trades) != 0 and self.state != State.RELOAD_CONFIG: | ||||||
|             msg = { |             msg = { | ||||||
| @@ -302,6 +302,15 @@ class FreqtradeBot(LoggingMixin): | |||||||
|                 self.update_trade_state(order.trade, order.order_id, fo, |                 self.update_trade_state(order.trade, order.order_id, fo, | ||||||
|                                         stoploss_order=(order.ft_order_side == 'stoploss')) |                                         stoploss_order=(order.ft_order_side == 'stoploss')) | ||||||
|  |  | ||||||
|  |             except InvalidOrderException as e: | ||||||
|  |                 logger.warning(f"Error updating Order {order.order_id} due to {e}.") | ||||||
|  |                 if order.order_date_utc - timedelta(days=5) < datetime.now(timezone.utc): | ||||||
|  |                     logger.warning( | ||||||
|  |                         "Order is older than 5 days. Assuming order was fully cancelled.") | ||||||
|  |                     fo = order.to_ccxt_object() | ||||||
|  |                     fo['status'] = 'canceled' | ||||||
|  |                     self.handle_timedout_order(fo, order.trade) | ||||||
|  |  | ||||||
|             except ExchangeError as e: |             except ExchangeError as e: | ||||||
|  |  | ||||||
|                 logger.warning(f"Error updating Order {order.order_id} due to {e}") |                 logger.warning(f"Error updating Order {order.order_id} due to {e}") | ||||||
| @@ -781,7 +790,7 @@ class FreqtradeBot(LoggingMixin): | |||||||
|                 current_rate=enter_limit_requested, |                 current_rate=enter_limit_requested, | ||||||
|                 proposed_leverage=1.0, |                 proposed_leverage=1.0, | ||||||
|                 max_leverage=max_leverage, |                 max_leverage=max_leverage, | ||||||
|                 side=trade_side, |                 side=trade_side, entry_tag=entry_tag, | ||||||
|             ) if self.trading_mode != TradingMode.SPOT else 1.0 |             ) if self.trading_mode != TradingMode.SPOT else 1.0 | ||||||
|             # Cap leverage between 1.0 and max_leverage. |             # Cap leverage between 1.0 and max_leverage. | ||||||
|             leverage = min(max(leverage, 1.0), max_leverage) |             leverage = min(max(leverage, 1.0), max_leverage) | ||||||
|   | |||||||
| @@ -704,7 +704,7 @@ class Backtesting: | |||||||
|                 current_rate=row[OPEN_IDX], |                 current_rate=row[OPEN_IDX], | ||||||
|                 proposed_leverage=1.0, |                 proposed_leverage=1.0, | ||||||
|                 max_leverage=max_leverage, |                 max_leverage=max_leverage, | ||||||
|                 side=direction, |                 side=direction, entry_tag=entry_tag, | ||||||
|             ) if self._can_short else 1.0 |             ) if self._can_short else 1.0 | ||||||
|             # Cap leverage between 1.0 and max_leverage. |             # Cap leverage between 1.0 and max_leverage. | ||||||
|             leverage = min(max(leverage, 1.0), max_leverage) |             leverage = min(max(leverage, 1.0), max_leverage) | ||||||
| @@ -966,6 +966,7 @@ class Backtesting: | |||||||
|                 return False |                 return False | ||||||
|             else: |             else: | ||||||
|                 del trade.orders[trade.orders.index(order)] |                 del trade.orders[trade.orders.index(order)] | ||||||
|  |                 trade.open_order_id = None | ||||||
|                 self.canceled_entry_orders += 1 |                 self.canceled_entry_orders += 1 | ||||||
|  |  | ||||||
|             # place new order if result was not None |             # place new order if result was not None | ||||||
| @@ -1094,6 +1095,7 @@ class Backtesting: | |||||||
|                     # 5. Process exit orders. |                     # 5. Process exit orders. | ||||||
|                     order = trade.select_order(trade.exit_side, is_open=True) |                     order = trade.select_order(trade.exit_side, is_open=True) | ||||||
|                     if order and self._get_order_filled(order.price, row): |                     if order and self._get_order_filled(order.price, row): | ||||||
|  |                         order.close_bt_order(current_time, trade) | ||||||
|                         trade.open_order_id = None |                         trade.open_order_id = None | ||||||
|                         trade.close_date = current_time |                         trade.close_date = current_time | ||||||
|                         trade.close(order.price, show_msg=False) |                         trade.close(order.price, show_msg=False) | ||||||
| @@ -1262,13 +1264,14 @@ class Backtesting: | |||||||
|                 self.results['strategy_comparison'].extend(results['strategy_comparison']) |                 self.results['strategy_comparison'].extend(results['strategy_comparison']) | ||||||
|             else: |             else: | ||||||
|                 self.results = results |                 self.results = results | ||||||
|  |             dt_appendix = datetime.now().strftime("%Y-%m-%d_%H-%M-%S") | ||||||
|             if self.config.get('export', 'none') in ('trades', 'signals'): |             if self.config.get('export', 'none') in ('trades', 'signals'): | ||||||
|                 store_backtest_stats(self.config['exportfilename'], self.results) |                 store_backtest_stats(self.config['exportfilename'], self.results, dt_appendix) | ||||||
|  |  | ||||||
|             if (self.config.get('export', 'none') == 'signals' and |             if (self.config.get('export', 'none') == 'signals' and | ||||||
|                     self.dataprovider.runmode == RunMode.BACKTEST): |                     self.dataprovider.runmode == RunMode.BACKTEST): | ||||||
|                 store_backtest_signal_candles(self.config['exportfilename'], self.processed_dfs) |                 store_backtest_signal_candles( | ||||||
|  |                     self.config['exportfilename'], self.processed_dfs, dt_appendix) | ||||||
|  |  | ||||||
|         # Results may be mixed up now. Sort them so they follow --strategy-list order. |         # Results may be mixed up now. Sort them so they follow --strategy-list order. | ||||||
|         if 'strategy_list' in self.config and len(self.results) > 0: |         if 'strategy_list' in self.config and len(self.results) > 0: | ||||||
|   | |||||||
| @@ -429,7 +429,7 @@ class Hyperopt: | |||||||
|             return new_list |             return new_list | ||||||
|         i = 0 |         i = 0 | ||||||
|         asked_non_tried: List[List[Any]] = [] |         asked_non_tried: List[List[Any]] = [] | ||||||
|         is_random: List[bool] = [] |         is_random_non_tried: List[bool] = [] | ||||||
|         while i < 5 and len(asked_non_tried) < n_points: |         while i < 5 and len(asked_non_tried) < n_points: | ||||||
|             if i < 3: |             if i < 3: | ||||||
|                 self.opt.cache_ = {} |                 self.opt.cache_ = {} | ||||||
| @@ -438,9 +438,9 @@ class Hyperopt: | |||||||
|             else: |             else: | ||||||
|                 asked = unique_list(self.opt.space.rvs(n_samples=n_points * 5)) |                 asked = unique_list(self.opt.space.rvs(n_samples=n_points * 5)) | ||||||
|                 is_random = [True for _ in range(len(asked))] |                 is_random = [True for _ in range(len(asked))] | ||||||
|             is_random += [rand for x, rand in zip(asked, is_random) |             is_random_non_tried += [rand for x, rand in zip(asked, is_random) | ||||||
|                           if x not in self.opt.Xi |                                     if x not in self.opt.Xi | ||||||
|                           and x not in asked_non_tried] |                                     and x not in asked_non_tried] | ||||||
|             asked_non_tried += [x for x in asked |             asked_non_tried += [x for x in asked | ||||||
|                                 if x not in self.opt.Xi |                                 if x not in self.opt.Xi | ||||||
|                                 and x not in asked_non_tried] |                                 and x not in asked_non_tried] | ||||||
| @@ -449,7 +449,7 @@ class Hyperopt: | |||||||
|         if asked_non_tried: |         if asked_non_tried: | ||||||
|             return ( |             return ( | ||||||
|                 asked_non_tried[:min(len(asked_non_tried), n_points)], |                 asked_non_tried[:min(len(asked_non_tried), n_points)], | ||||||
|                 is_random[:min(len(asked_non_tried), n_points)] |                 is_random_non_tried[:min(len(asked_non_tried), n_points)] | ||||||
|             ) |             ) | ||||||
|         else: |         else: | ||||||
|             return self.opt.ask(n_points=n_points), [False for _ in range(n_points)] |             return self.opt.ask(n_points=n_points), [False for _ in range(n_points)] | ||||||
|   | |||||||
| @@ -4,7 +4,6 @@ from datetime import datetime, timedelta, timezone | |||||||
| from pathlib import Path | from pathlib import Path | ||||||
| from typing import Any, Dict, List, Union | from typing import Any, Dict, List, Union | ||||||
|  |  | ||||||
| from numpy import int64 |  | ||||||
| from pandas import DataFrame, to_datetime | from pandas import DataFrame, to_datetime | ||||||
| from tabulate import tabulate | from tabulate import tabulate | ||||||
|  |  | ||||||
| @@ -18,21 +17,21 @@ from freqtrade.optimize.backtest_caching import get_backtest_metadata_filename | |||||||
| logger = logging.getLogger(__name__) | logger = logging.getLogger(__name__) | ||||||
|  |  | ||||||
|  |  | ||||||
| def store_backtest_stats(recordfilename: Path, stats: Dict[str, DataFrame]) -> None: | def store_backtest_stats( | ||||||
|  |         recordfilename: Path, stats: Dict[str, DataFrame], dtappendix: str) -> None: | ||||||
|     """ |     """ | ||||||
|     Stores backtest results |     Stores backtest results | ||||||
|     :param recordfilename: Path object, which can either be a filename or a directory. |     :param recordfilename: Path object, which can either be a filename or a directory. | ||||||
|         Filenames will be appended with a timestamp right before the suffix |         Filenames will be appended with a timestamp right before the suffix | ||||||
|         while for directories, <directory>/backtest-result-<datetime>.json will be used as filename |         while for directories, <directory>/backtest-result-<datetime>.json will be used as filename | ||||||
|     :param stats: Dataframe containing the backtesting statistics |     :param stats: Dataframe containing the backtesting statistics | ||||||
|  |     :param dtappendix: Datetime to use for the filename | ||||||
|     """ |     """ | ||||||
|     if recordfilename.is_dir(): |     if recordfilename.is_dir(): | ||||||
|         filename = (recordfilename / |         filename = (recordfilename / f'backtest-result-{dtappendix}.json') | ||||||
|                     f'backtest-result-{datetime.now().strftime("%Y-%m-%d_%H-%M-%S")}.json') |  | ||||||
|     else: |     else: | ||||||
|         filename = Path.joinpath( |         filename = Path.joinpath( | ||||||
|             recordfilename.parent, |             recordfilename.parent, f'{recordfilename.stem}-{dtappendix}' | ||||||
|             f'{recordfilename.stem}-{datetime.now().strftime("%Y-%m-%d_%H-%M-%S")}' |  | ||||||
|         ).with_suffix(recordfilename.suffix) |         ).with_suffix(recordfilename.suffix) | ||||||
|  |  | ||||||
|     # Store metadata separately. |     # Store metadata separately. | ||||||
| @@ -45,7 +44,8 @@ def store_backtest_stats(recordfilename: Path, stats: Dict[str, DataFrame]) -> N | |||||||
|     file_dump_json(latest_filename, {'latest_backtest': str(filename.name)}) |     file_dump_json(latest_filename, {'latest_backtest': str(filename.name)}) | ||||||
|  |  | ||||||
|  |  | ||||||
| def store_backtest_signal_candles(recordfilename: Path, candles: Dict[str, Dict]) -> Path: | def store_backtest_signal_candles( | ||||||
|  |         recordfilename: Path, candles: Dict[str, Dict], dtappendix: str) -> Path: | ||||||
|     """ |     """ | ||||||
|     Stores backtest trade signal candles |     Stores backtest trade signal candles | ||||||
|     :param recordfilename: Path object, which can either be a filename or a directory. |     :param recordfilename: Path object, which can either be a filename or a directory. | ||||||
| @@ -53,14 +53,13 @@ def store_backtest_signal_candles(recordfilename: Path, candles: Dict[str, Dict] | |||||||
|         while for directories, <directory>/backtest-result-<datetime>_signals.pkl will be used |         while for directories, <directory>/backtest-result-<datetime>_signals.pkl will be used | ||||||
|         as filename |         as filename | ||||||
|     :param stats: Dict containing the backtesting signal candles |     :param stats: Dict containing the backtesting signal candles | ||||||
|  |     :param dtappendix: Datetime to use for the filename | ||||||
|     """ |     """ | ||||||
|     if recordfilename.is_dir(): |     if recordfilename.is_dir(): | ||||||
|         filename = (recordfilename / |         filename = (recordfilename / f'backtest-result-{dtappendix}_signals.pkl') | ||||||
|                     f'backtest-result-{datetime.now().strftime("%Y-%m-%d_%H-%M-%S")}_signals.pkl') |  | ||||||
|     else: |     else: | ||||||
|         filename = Path.joinpath( |         filename = Path.joinpath( | ||||||
|             recordfilename.parent, |             recordfilename.parent, f'{recordfilename.stem}-{dtappendix}_signals.pkl' | ||||||
|             f'{recordfilename.stem}-{datetime.now().strftime("%Y-%m-%d_%H-%M-%S")}_signals.pkl' |  | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     file_dump_joblib(filename, candles) |     file_dump_joblib(filename, candles) | ||||||
| @@ -417,9 +416,6 @@ def generate_strategy_stats(pairlist: List[str], | |||||||
|                     key=lambda x: x['profit_sum']) if len(pair_results) > 1 else None |                     key=lambda x: x['profit_sum']) if len(pair_results) > 1 else None | ||||||
|     worst_pair = min([pair for pair in pair_results if pair['key'] != 'TOTAL'], |     worst_pair = min([pair for pair in pair_results if pair['key'] != 'TOTAL'], | ||||||
|                      key=lambda x: x['profit_sum']) if len(pair_results) > 1 else None |                      key=lambda x: x['profit_sum']) if len(pair_results) > 1 else None | ||||||
|     if not results.empty: |  | ||||||
|         results['open_timestamp'] = results['open_date'].view(int64) // 1e6 |  | ||||||
|         results['close_timestamp'] = results['close_date'].view(int64) // 1e6 |  | ||||||
|  |  | ||||||
|     backtest_days = (max_date - min_date).days or 1 |     backtest_days = (max_date - min_date).days or 1 | ||||||
|     strat_stats = { |     strat_stats = { | ||||||
|   | |||||||
| @@ -247,6 +247,35 @@ def set_sqlite_to_wal(engine): | |||||||
|             connection.execute(text("PRAGMA journal_mode=wal")) |             connection.execute(text("PRAGMA journal_mode=wal")) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def fix_old_dry_orders(engine): | ||||||
|  |     with engine.begin() as connection: | ||||||
|  |         connection.execute( | ||||||
|  |             text( | ||||||
|  |                 """ | ||||||
|  |                 update orders | ||||||
|  |                 set ft_is_open = 0 | ||||||
|  |                 where ft_is_open = 1 and (ft_trade_id, order_id) not in ( | ||||||
|  |                     select id, stoploss_order_id from trades where stoploss_order_id is not null | ||||||
|  |                 ) and ft_order_side = 'stoploss' | ||||||
|  |                 and order_id like 'dry_%' | ||||||
|  |                 """ | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         connection.execute( | ||||||
|  |             text( | ||||||
|  |                 """ | ||||||
|  |                 update orders | ||||||
|  |                 set ft_is_open = 0 | ||||||
|  |                 where ft_is_open = 1 | ||||||
|  |                 and (ft_trade_id, order_id) not in ( | ||||||
|  |                     select id, open_order_id from trades where open_order_id is not null | ||||||
|  |                 ) and ft_order_side != 'stoploss' | ||||||
|  |                 and order_id like 'dry_%' | ||||||
|  |                 """ | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
| def check_migrate(engine, decl_base, previous_tables) -> None: | def check_migrate(engine, decl_base, previous_tables) -> None: | ||||||
|     """ |     """ | ||||||
|     Checks if migration is necessary and migrates if necessary |     Checks if migration is necessary and migrates if necessary | ||||||
| @@ -288,3 +317,4 @@ def check_migrate(engine, decl_base, previous_tables) -> None: | |||||||
|             "start with a fresh database.") |             "start with a fresh database.") | ||||||
|  |  | ||||||
|     set_sqlite_to_wal(engine) |     set_sqlite_to_wal(engine) | ||||||
|  |     fix_old_dry_orders(engine) | ||||||
|   | |||||||
| @@ -74,7 +74,7 @@ class Order(_DECL_BASE): | |||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def safe_filled(self) -> float: |     def safe_filled(self) -> float: | ||||||
|         return self.filled or self.amount or 0.0 |         return self.filled if self.filled is not None else self.amount or 0.0 | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def safe_fee_base(self) -> float: |     def safe_fee_base(self) -> float: | ||||||
| @@ -137,35 +137,40 @@ class Order(_DECL_BASE): | |||||||
|             'info': {}, |             'info': {}, | ||||||
|         } |         } | ||||||
|  |  | ||||||
|     def to_json(self, entry_side: str) -> Dict[str, Any]: |     def to_json(self, entry_side: str, minified: bool = False) -> Dict[str, Any]: | ||||||
|         return { |         resp = { | ||||||
|             'pair': self.ft_pair, |  | ||||||
|             'order_id': self.order_id, |  | ||||||
|             'status': self.status, |  | ||||||
|             'amount': self.amount, |             'amount': self.amount, | ||||||
|             'average': round(self.average, 8) if self.average else 0, |  | ||||||
|             'safe_price': self.safe_price, |             'safe_price': self.safe_price, | ||||||
|             'cost': self.cost if self.cost else 0, |  | ||||||
|             'filled': self.filled, |  | ||||||
|             'ft_order_side': self.ft_order_side, |             'ft_order_side': self.ft_order_side, | ||||||
|             'is_open': self.ft_is_open, |  | ||||||
|             'order_date': self.order_date.strftime(DATETIME_PRINT_FORMAT) |  | ||||||
|             if self.order_date else None, |  | ||||||
|             'order_timestamp': int(self.order_date.replace( |  | ||||||
|                 tzinfo=timezone.utc).timestamp() * 1000) if self.order_date else None, |  | ||||||
|             'order_filled_date': self.order_filled_date.strftime(DATETIME_PRINT_FORMAT) |  | ||||||
|             if self.order_filled_date else None, |  | ||||||
|             'order_filled_timestamp': int(self.order_filled_date.replace( |             'order_filled_timestamp': int(self.order_filled_date.replace( | ||||||
|                 tzinfo=timezone.utc).timestamp() * 1000) if self.order_filled_date else None, |                 tzinfo=timezone.utc).timestamp() * 1000) if self.order_filled_date else None, | ||||||
|             'order_type': self.order_type, |  | ||||||
|             'price': self.price, |  | ||||||
|             'ft_is_entry': self.ft_order_side == entry_side, |             'ft_is_entry': self.ft_order_side == entry_side, | ||||||
|             'remaining': self.remaining, |  | ||||||
|         } |         } | ||||||
|  |         if not minified: | ||||||
|  |             resp.update({ | ||||||
|  |                 'pair': self.ft_pair, | ||||||
|  |                 'order_id': self.order_id, | ||||||
|  |                 'status': self.status, | ||||||
|  |                 'average': round(self.average, 8) if self.average else 0, | ||||||
|  |                 'cost': self.cost if self.cost else 0, | ||||||
|  |                 'filled': self.filled, | ||||||
|  |                 'is_open': self.ft_is_open, | ||||||
|  |                 'order_date': self.order_date.strftime(DATETIME_PRINT_FORMAT) | ||||||
|  |                 if self.order_date else None, | ||||||
|  |                 'order_timestamp': int(self.order_date.replace( | ||||||
|  |                     tzinfo=timezone.utc).timestamp() * 1000) if self.order_date else None, | ||||||
|  |                 'order_filled_date': self.order_filled_date.strftime(DATETIME_PRINT_FORMAT) | ||||||
|  |                 if self.order_filled_date else None, | ||||||
|  |                 'order_type': self.order_type, | ||||||
|  |                 'price': self.price, | ||||||
|  |                 'remaining': self.remaining, | ||||||
|  |             }) | ||||||
|  |         return resp | ||||||
|  |  | ||||||
|     def close_bt_order(self, close_date: datetime, trade: 'LocalTrade'): |     def close_bt_order(self, close_date: datetime, trade: 'LocalTrade'): | ||||||
|         self.order_filled_date = close_date |         self.order_filled_date = close_date | ||||||
|         self.filled = self.amount |         self.filled = self.amount | ||||||
|  |         self.remaining = 0 | ||||||
|         self.status = 'closed' |         self.status = 'closed' | ||||||
|         self.ft_is_open = False |         self.ft_is_open = False | ||||||
|         if (self.ft_order_side == trade.entry_side |         if (self.ft_order_side == trade.entry_side | ||||||
| @@ -393,9 +398,9 @@ class LocalTrade(): | |||||||
|             f'open_rate={self.open_rate:.8f}, open_since={open_since})' |             f'open_rate={self.open_rate:.8f}, open_since={open_since})' | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     def to_json(self) -> Dict[str, Any]: |     def to_json(self, minified: bool = False) -> Dict[str, Any]: | ||||||
|         filled_orders = self.select_filled_orders() |         filled_orders = self.select_filled_or_open_orders() | ||||||
|         orders = [order.to_json(self.entry_side) for order in filled_orders] |         orders = [order.to_json(self.entry_side, minified) for order in filled_orders] | ||||||
|  |  | ||||||
|         return { |         return { | ||||||
|             'trade_id': self.id, |             'trade_id': self.id, | ||||||
| @@ -823,14 +828,6 @@ class LocalTrade(): | |||||||
|         return float(f"{profit_ratio:.8f}") |         return float(f"{profit_ratio:.8f}") | ||||||
|  |  | ||||||
|     def recalc_trade_from_orders(self): |     def recalc_trade_from_orders(self): | ||||||
|         # We need at least 2 entry orders for averaging amounts and rates. |  | ||||||
|         # TODO: this condition could probably be removed |  | ||||||
|         if len(self.select_filled_orders(self.entry_side)) < 2: |  | ||||||
|             self.stake_amount = self.amount * self.open_rate / self.leverage |  | ||||||
|  |  | ||||||
|             # Just in case, still recalc open trade value |  | ||||||
|             self.recalc_open_trade_value() |  | ||||||
|             return |  | ||||||
|  |  | ||||||
|         total_amount = 0.0 |         total_amount = 0.0 | ||||||
|         total_stake = 0.0 |         total_stake = 0.0 | ||||||
| @@ -842,8 +839,6 @@ class LocalTrade(): | |||||||
|  |  | ||||||
|             tmp_amount = o.safe_amount_after_fee |             tmp_amount = o.safe_amount_after_fee | ||||||
|             tmp_price = o.average or o.price |             tmp_price = o.average or o.price | ||||||
|             if o.filled is not None: |  | ||||||
|                 tmp_amount = o.filled |  | ||||||
|             if tmp_amount > 0.0 and tmp_price is not None: |             if tmp_amount > 0.0 and tmp_price is not None: | ||||||
|                 total_amount += tmp_amount |                 total_amount += tmp_amount | ||||||
|                 total_stake += tmp_price * tmp_amount |                 total_stake += tmp_price * tmp_amount | ||||||
| @@ -897,6 +892,21 @@ class LocalTrade(): | |||||||
|                 (o.filled or 0) > 0 and |                 (o.filled or 0) > 0 and | ||||||
|                 o.status in NON_OPEN_EXCHANGE_STATES] |                 o.status in NON_OPEN_EXCHANGE_STATES] | ||||||
|  |  | ||||||
|  |     def select_filled_or_open_orders(self) -> List['Order']: | ||||||
|  |         """ | ||||||
|  |         Finds filled or open orders | ||||||
|  |         :param order_side: Side of the order (either 'buy', 'sell', or None) | ||||||
|  |         :return: array of Order objects | ||||||
|  |         """ | ||||||
|  |         return [o for o in self.orders if | ||||||
|  |                 ( | ||||||
|  |                     o.ft_is_open is False | ||||||
|  |                     and (o.filled or 0) > 0 | ||||||
|  |                     and o.status in NON_OPEN_EXCHANGE_STATES | ||||||
|  |                     ) | ||||||
|  |                 or (o.ft_is_open is True and o.status is not None) | ||||||
|  |                 ] | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def nr_of_successful_entries(self) -> int: |     def nr_of_successful_entries(self) -> int: | ||||||
|         """ |         """ | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| import asyncio | import asyncio | ||||||
| import logging | import logging | ||||||
| from copy import deepcopy | from copy import deepcopy | ||||||
|  | from datetime import datetime | ||||||
| from typing import Any, Dict, List | from typing import Any, Dict, List | ||||||
|  |  | ||||||
| from fastapi import APIRouter, BackgroundTasks, Depends | from fastapi import APIRouter, BackgroundTasks, Depends | ||||||
| @@ -102,7 +103,10 @@ async def api_start_backtest(bt_settings: BacktestRequest, background_tasks: Bac | |||||||
|                     min_date=min_date, max_date=max_date) |                     min_date=min_date, max_date=max_date) | ||||||
|  |  | ||||||
|             if btconfig.get('export', 'none') == 'trades': |             if btconfig.get('export', 'none') == 'trades': | ||||||
|                 store_backtest_stats(btconfig['exportfilename'], ApiServer._bt.results) |                 store_backtest_stats( | ||||||
|  |                     btconfig['exportfilename'], ApiServer._bt.results, | ||||||
|  |                     datetime.now().strftime("%Y-%m-%d_%H-%M-%S") | ||||||
|  |                     ) | ||||||
|  |  | ||||||
|             logger.info("Backtest finished.") |             logger.info("Backtest finished.") | ||||||
|  |  | ||||||
|   | |||||||
| @@ -120,6 +120,8 @@ class Stats(BaseModel): | |||||||
| class DailyRecord(BaseModel): | class DailyRecord(BaseModel): | ||||||
|     date: date |     date: date | ||||||
|     abs_profit: float |     abs_profit: float | ||||||
|  |     rel_profit: float | ||||||
|  |     starting_balance: float | ||||||
|     fiat_value: float |     fiat_value: float | ||||||
|     trade_count: int |     trade_count: int | ||||||
|  |  | ||||||
| @@ -166,7 +168,7 @@ class ShowConfig(BaseModel): | |||||||
|     trailing_stop_positive: Optional[float] |     trailing_stop_positive: Optional[float] | ||||||
|     trailing_stop_positive_offset: Optional[float] |     trailing_stop_positive_offset: Optional[float] | ||||||
|     trailing_only_offset_is_reached: Optional[bool] |     trailing_only_offset_is_reached: Optional[bool] | ||||||
|     unfilledtimeout: UnfilledTimeout |     unfilledtimeout: Optional[UnfilledTimeout]  # Empty in webserver mode | ||||||
|     order_types: Optional[OrderTypes] |     order_types: Optional[OrderTypes] | ||||||
|     use_custom_stoploss: Optional[bool] |     use_custom_stoploss: Optional[bool] | ||||||
|     timeframe: Optional[str] |     timeframe: Optional[str] | ||||||
|   | |||||||
| @@ -36,7 +36,8 @@ logger = logging.getLogger(__name__) | |||||||
| # versions 2.xx -> futures/short branch | # versions 2.xx -> futures/short branch | ||||||
| # 2.14: Add entry/exit orders to trade response | # 2.14: Add entry/exit orders to trade response | ||||||
| # 2.15: Add backtest history endpoints | # 2.15: Add backtest history endpoints | ||||||
| API_VERSION = 2.15 | # 2.16: Additional daily metrics | ||||||
|  | API_VERSION = 2.16 | ||||||
|  |  | ||||||
| # Public API, requires no auth. | # Public API, requires no auth. | ||||||
| router_public = APIRouter() | router_public = APIRouter() | ||||||
| @@ -86,8 +87,8 @@ def stats(rpc: RPC = Depends(get_rpc)): | |||||||
|  |  | ||||||
| @router.get('/daily', response_model=Daily, tags=['info']) | @router.get('/daily', response_model=Daily, tags=['info']) | ||||||
| def daily(timescale: int = 7, rpc: RPC = Depends(get_rpc), config=Depends(get_config)): | def daily(timescale: int = 7, rpc: RPC = Depends(get_rpc), config=Depends(get_config)): | ||||||
|     return rpc._rpc_daily_profit(timescale, config['stake_currency'], |     return rpc._rpc_timeunit_profit(timescale, config['stake_currency'], | ||||||
|                                  config.get('fiat_display_currency', '')) |                                     config.get('fiat_display_currency', '')) | ||||||
|  |  | ||||||
|  |  | ||||||
| @router.get('/status', response_model=List[OpenTradeSchema], tags=['info']) | @router.get('/status', response_model=List[OpenTradeSchema], tags=['info']) | ||||||
|   | |||||||
							
								
								
									
										59
									
								
								freqtrade/rpc/discord.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								freqtrade/rpc/discord.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | |||||||
|  | import logging | ||||||
|  | from typing import Any, Dict | ||||||
|  |  | ||||||
|  | from freqtrade.enums.rpcmessagetype import RPCMessageType | ||||||
|  | from freqtrade.rpc import RPC | ||||||
|  | from freqtrade.rpc.webhook import Webhook | ||||||
|  |  | ||||||
|  |  | ||||||
|  | logger = logging.getLogger(__name__) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Discord(Webhook): | ||||||
|  |     def __init__(self, rpc: 'RPC', config: Dict[str, Any]): | ||||||
|  |         # super().__init__(rpc, config) | ||||||
|  |         self.rpc = rpc | ||||||
|  |         self.config = config | ||||||
|  |         self.strategy = config.get('strategy', '') | ||||||
|  |         self.timeframe = config.get('timeframe', '') | ||||||
|  |  | ||||||
|  |         self._url = self.config['discord']['webhook_url'] | ||||||
|  |         self._format = 'json' | ||||||
|  |         self._retries = 1 | ||||||
|  |         self._retry_delay = 0.1 | ||||||
|  |  | ||||||
|  |     def cleanup(self) -> None: | ||||||
|  |         """ | ||||||
|  |         Cleanup pending module resources. | ||||||
|  |         This will do nothing for webhooks, they will simply not be called anymore | ||||||
|  |         """ | ||||||
|  |         pass | ||||||
|  |  | ||||||
|  |     def send_msg(self, msg) -> None: | ||||||
|  |         logger.info(f"Sending discord message: {msg}") | ||||||
|  |  | ||||||
|  |         if msg['type'].value in self.config['discord']: | ||||||
|  |  | ||||||
|  |             msg['strategy'] = self.strategy | ||||||
|  |             msg['timeframe'] = self.timeframe | ||||||
|  |             fields = self.config['discord'].get(msg['type'].value) | ||||||
|  |             color = 0x0000FF | ||||||
|  |             if msg['type'] in (RPCMessageType.EXIT, RPCMessageType.EXIT_FILL): | ||||||
|  |                 profit_ratio = msg.get('profit_ratio') | ||||||
|  |                 color = (0x00FF00 if profit_ratio > 0 else 0xFF0000) | ||||||
|  |  | ||||||
|  |             embeds = [{ | ||||||
|  |                 'title': f"Trade: {msg['pair']} {msg['type'].value}", | ||||||
|  |                 'color': color, | ||||||
|  |                 'fields': [], | ||||||
|  |  | ||||||
|  |             }] | ||||||
|  |             for f in fields: | ||||||
|  |                 for k, v in f.items(): | ||||||
|  |                     v = v.format(**msg) | ||||||
|  |                     embeds[0]['fields'].append(  # type: ignore | ||||||
|  |                         {'name': k, 'value': v, 'inline': True}) | ||||||
|  |  | ||||||
|  |             # Send the message to discord channel | ||||||
|  |             payload = {'embeds': embeds} | ||||||
|  |             self._send_msg(payload) | ||||||
| @@ -283,33 +283,57 @@ class RPC: | |||||||
|                 columns.append('# Entries') |                 columns.append('# Entries') | ||||||
|             return trades_list, columns, fiat_profit_sum |             return trades_list, columns, fiat_profit_sum | ||||||
|  |  | ||||||
|     def _rpc_daily_profit( |     def _rpc_timeunit_profit( | ||||||
|             self, timescale: int, |             self, timescale: int, | ||||||
|             stake_currency: str, fiat_display_currency: str) -> Dict[str, Any]: |             stake_currency: str, fiat_display_currency: str, | ||||||
|         today = datetime.now(timezone.utc).date() |             timeunit: str = 'days') -> Dict[str, Any]: | ||||||
|         profit_days: Dict[date, Dict] = {} |         """ | ||||||
|  |         :param timeunit: Valid entries are 'days', 'weeks', 'months' | ||||||
|  |         """ | ||||||
|  |         start_date = datetime.now(timezone.utc).date() | ||||||
|  |         if timeunit == 'weeks': | ||||||
|  |             # weekly | ||||||
|  |             start_date = start_date - timedelta(days=start_date.weekday())  # Monday | ||||||
|  |         if timeunit == 'months': | ||||||
|  |             start_date = start_date.replace(day=1) | ||||||
|  |  | ||||||
|  |         def time_offset(step: int): | ||||||
|  |             if timeunit == 'months': | ||||||
|  |                 return relativedelta(months=step) | ||||||
|  |             return timedelta(**{timeunit: step}) | ||||||
|  |  | ||||||
|         if not (isinstance(timescale, int) and timescale > 0): |         if not (isinstance(timescale, int) and timescale > 0): | ||||||
|             raise RPCException('timescale must be an integer greater than 0') |             raise RPCException('timescale must be an integer greater than 0') | ||||||
|  |  | ||||||
|  |         profit_units: Dict[date, Dict] = {} | ||||||
|  |         daily_stake = self._freqtrade.wallets.get_total_stake_amount() | ||||||
|  |  | ||||||
|         for day in range(0, timescale): |         for day in range(0, timescale): | ||||||
|             profitday = today - timedelta(days=day) |             profitday = start_date - time_offset(day) | ||||||
|             trades = Trade.get_trades(trade_filter=[ |             # Only query for necessary columns for performance reasons. | ||||||
|  |             trades = Trade.query.session.query(Trade.close_profit_abs).filter( | ||||||
|                 Trade.is_open.is_(False), |                 Trade.is_open.is_(False), | ||||||
|                 Trade.close_date >= profitday, |                 Trade.close_date >= profitday, | ||||||
|                 Trade.close_date < (profitday + timedelta(days=1)) |                 Trade.close_date < (profitday + time_offset(1)) | ||||||
|             ]).order_by(Trade.close_date).all() |             ).order_by(Trade.close_date).all() | ||||||
|  |  | ||||||
|             curdayprofit = sum( |             curdayprofit = sum( | ||||||
|                 trade.close_profit_abs for trade in trades if trade.close_profit_abs is not None) |                 trade.close_profit_abs for trade in trades if trade.close_profit_abs is not None) | ||||||
|             profit_days[profitday] = { |             # Calculate this periods starting balance | ||||||
|  |             daily_stake = daily_stake - curdayprofit | ||||||
|  |             profit_units[profitday] = { | ||||||
|                 'amount': curdayprofit, |                 'amount': curdayprofit, | ||||||
|                 'trades': len(trades) |                 'daily_stake': daily_stake, | ||||||
|  |                 'rel_profit': round(curdayprofit / daily_stake, 8) if daily_stake > 0 else 0, | ||||||
|  |                 'trades': len(trades), | ||||||
|             } |             } | ||||||
|  |  | ||||||
|         data = [ |         data = [ | ||||||
|             { |             { | ||||||
|                 'date': key, |                 'date': f"{key.year}-{key.month:02d}" if timeunit == 'months' else key, | ||||||
|                 'abs_profit': value["amount"], |                 'abs_profit': value["amount"], | ||||||
|  |                 'starting_balance': value["daily_stake"], | ||||||
|  |                 'rel_profit': value["rel_profit"], | ||||||
|                 'fiat_value': self._fiat_converter.convert_amount( |                 'fiat_value': self._fiat_converter.convert_amount( | ||||||
|                     value['amount'], |                     value['amount'], | ||||||
|                     stake_currency, |                     stake_currency, | ||||||
| @@ -317,92 +341,7 @@ class RPC: | |||||||
|                 ) if self._fiat_converter else 0, |                 ) if self._fiat_converter else 0, | ||||||
|                 'trade_count': value["trades"], |                 'trade_count': value["trades"], | ||||||
|             } |             } | ||||||
|             for key, value in profit_days.items() |             for key, value in profit_units.items() | ||||||
|         ] |  | ||||||
|         return { |  | ||||||
|             'stake_currency': stake_currency, |  | ||||||
|             'fiat_display_currency': fiat_display_currency, |  | ||||||
|             'data': data |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|     def _rpc_weekly_profit( |  | ||||||
|             self, timescale: int, |  | ||||||
|             stake_currency: str, fiat_display_currency: str) -> Dict[str, Any]: |  | ||||||
|         today = datetime.now(timezone.utc).date() |  | ||||||
|         first_iso_day_of_week = today - timedelta(days=today.weekday())  # Monday |  | ||||||
|         profit_weeks: Dict[date, Dict] = {} |  | ||||||
|  |  | ||||||
|         if not (isinstance(timescale, int) and timescale > 0): |  | ||||||
|             raise RPCException('timescale must be an integer greater than 0') |  | ||||||
|  |  | ||||||
|         for week in range(0, timescale): |  | ||||||
|             profitweek = first_iso_day_of_week - timedelta(weeks=week) |  | ||||||
|             trades = Trade.get_trades(trade_filter=[ |  | ||||||
|                 Trade.is_open.is_(False), |  | ||||||
|                 Trade.close_date >= profitweek, |  | ||||||
|                 Trade.close_date < (profitweek + timedelta(weeks=1)) |  | ||||||
|             ]).order_by(Trade.close_date).all() |  | ||||||
|             curweekprofit = sum( |  | ||||||
|                 trade.close_profit_abs for trade in trades if trade.close_profit_abs is not None) |  | ||||||
|             profit_weeks[profitweek] = { |  | ||||||
|                 'amount': curweekprofit, |  | ||||||
|                 'trades': len(trades) |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|         data = [ |  | ||||||
|             { |  | ||||||
|                 'date': key, |  | ||||||
|                 'abs_profit': value["amount"], |  | ||||||
|                 'fiat_value': self._fiat_converter.convert_amount( |  | ||||||
|                     value['amount'], |  | ||||||
|                     stake_currency, |  | ||||||
|                     fiat_display_currency |  | ||||||
|                 ) if self._fiat_converter else 0, |  | ||||||
|                 'trade_count': value["trades"], |  | ||||||
|             } |  | ||||||
|             for key, value in profit_weeks.items() |  | ||||||
|         ] |  | ||||||
|         return { |  | ||||||
|             'stake_currency': stake_currency, |  | ||||||
|             'fiat_display_currency': fiat_display_currency, |  | ||||||
|             'data': data |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|     def _rpc_monthly_profit( |  | ||||||
|             self, timescale: int, |  | ||||||
|             stake_currency: str, fiat_display_currency: str) -> Dict[str, Any]: |  | ||||||
|         first_day_of_month = datetime.now(timezone.utc).date().replace(day=1) |  | ||||||
|         profit_months: Dict[date, Dict] = {} |  | ||||||
|  |  | ||||||
|         if not (isinstance(timescale, int) and timescale > 0): |  | ||||||
|             raise RPCException('timescale must be an integer greater than 0') |  | ||||||
|  |  | ||||||
|         for month in range(0, timescale): |  | ||||||
|             profitmonth = first_day_of_month - relativedelta(months=month) |  | ||||||
|             trades = Trade.get_trades(trade_filter=[ |  | ||||||
|                 Trade.is_open.is_(False), |  | ||||||
|                 Trade.close_date >= profitmonth, |  | ||||||
|                 Trade.close_date < (profitmonth + relativedelta(months=1)) |  | ||||||
|             ]).order_by(Trade.close_date).all() |  | ||||||
|             curmonthprofit = sum( |  | ||||||
|                 trade.close_profit_abs for trade in trades if trade.close_profit_abs is not None) |  | ||||||
|             profit_months[profitmonth] = { |  | ||||||
|                 'amount': curmonthprofit, |  | ||||||
|                 'trades': len(trades) |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|         data = [ |  | ||||||
|             { |  | ||||||
|                 'date': f"{key.year}-{key.month:02d}", |  | ||||||
|                 'abs_profit': value["amount"], |  | ||||||
|                 'fiat_value': self._fiat_converter.convert_amount( |  | ||||||
|                     value['amount'], |  | ||||||
|                     stake_currency, |  | ||||||
|                     fiat_display_currency |  | ||||||
|                 ) if self._fiat_converter else 0, |  | ||||||
|                 'trade_count': value["trades"], |  | ||||||
|             } |  | ||||||
|             for key, value in profit_months.items() |  | ||||||
|         ] |         ] | ||||||
|         return { |         return { | ||||||
|             'stake_currency': stake_currency, |             'stake_currency': stake_currency, | ||||||
|   | |||||||
| @@ -27,6 +27,12 @@ class RPCManager: | |||||||
|             from freqtrade.rpc.telegram import Telegram |             from freqtrade.rpc.telegram import Telegram | ||||||
|             self.registered_modules.append(Telegram(self._rpc, config)) |             self.registered_modules.append(Telegram(self._rpc, config)) | ||||||
|  |  | ||||||
|  |         # Enable discord | ||||||
|  |         if config.get('discord', {}).get('enabled', False): | ||||||
|  |             logger.info('Enabling rpc.discord ...') | ||||||
|  |             from freqtrade.rpc.discord import Discord | ||||||
|  |             self.registered_modules.append(Discord(self._rpc, config)) | ||||||
|  |  | ||||||
|         # Enable Webhook |         # Enable Webhook | ||||||
|         if config.get('webhook', {}).get('enabled', False): |         if config.get('webhook', {}).get('enabled', False): | ||||||
|             logger.info('Enabling rpc.webhook ...') |             logger.info('Enabling rpc.webhook ...') | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ This module manage Telegram communication | |||||||
| import json | import json | ||||||
| import logging | import logging | ||||||
| import re | import re | ||||||
|  | from dataclasses import dataclass | ||||||
| from datetime import date, datetime, timedelta | from datetime import date, datetime, timedelta | ||||||
| from functools import partial | from functools import partial | ||||||
| from html import escape | from html import escape | ||||||
| @@ -37,6 +38,15 @@ logger.debug('Included module rpc.telegram ...') | |||||||
| MAX_TELEGRAM_MESSAGE_LENGTH = 4096 | MAX_TELEGRAM_MESSAGE_LENGTH = 4096 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass | ||||||
|  | class TimeunitMappings: | ||||||
|  |     header: str | ||||||
|  |     message: str | ||||||
|  |     message2: str | ||||||
|  |     callback: str | ||||||
|  |     default: int | ||||||
|  |  | ||||||
|  |  | ||||||
| def authorized_only(command_handler: Callable[..., None]) -> Callable[..., Any]: | def authorized_only(command_handler: Callable[..., None]) -> Callable[..., Any]: | ||||||
|     """ |     """ | ||||||
|     Decorator to check if the message comes from the correct chat_id |     Decorator to check if the message comes from the correct chat_id | ||||||
| @@ -404,7 +414,7 @@ class Telegram(RPCHandler): | |||||||
|             first_avg = filled_orders[0]["safe_price"] |             first_avg = filled_orders[0]["safe_price"] | ||||||
|  |  | ||||||
|         for x, order in enumerate(filled_orders): |         for x, order in enumerate(filled_orders): | ||||||
|             if not order['ft_is_entry']: |             if not order['ft_is_entry'] or order['is_open'] is True: | ||||||
|                 continue |                 continue | ||||||
|             cur_entry_datetime = arrow.get(order["order_filled_date"]) |             cur_entry_datetime = arrow.get(order["order_filled_date"]) | ||||||
|             cur_entry_amount = order["amount"] |             cur_entry_amount = order["amount"] | ||||||
| @@ -571,6 +581,60 @@ class Telegram(RPCHandler): | |||||||
|         except RPCException as e: |         except RPCException as e: | ||||||
|             self._send_msg(str(e)) |             self._send_msg(str(e)) | ||||||
|  |  | ||||||
|  |     @authorized_only | ||||||
|  |     def _timeunit_stats(self, update: Update, context: CallbackContext, unit: str) -> None: | ||||||
|  |         """ | ||||||
|  |         Handler for /daily <n> | ||||||
|  |         Returns a daily profit (in BTC) over the last n days. | ||||||
|  |         :param bot: telegram bot | ||||||
|  |         :param update: message update | ||||||
|  |         :return: None | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         vals = { | ||||||
|  |             'days': TimeunitMappings('Day', 'Daily', 'days', 'update_daily', 7), | ||||||
|  |             'weeks': TimeunitMappings('Monday', 'Weekly', 'weeks (starting from Monday)', | ||||||
|  |                                       'update_weekly', 8), | ||||||
|  |             'months': TimeunitMappings('Month', 'Monthly', 'months', 'update_monthly', 6), | ||||||
|  |         } | ||||||
|  |         val = vals[unit] | ||||||
|  |  | ||||||
|  |         stake_cur = self._config['stake_currency'] | ||||||
|  |         fiat_disp_cur = self._config.get('fiat_display_currency', '') | ||||||
|  |         try: | ||||||
|  |             timescale = int(context.args[0]) if context.args else val.default | ||||||
|  |         except (TypeError, ValueError, IndexError): | ||||||
|  |             timescale = val.default | ||||||
|  |         try: | ||||||
|  |             stats = self._rpc._rpc_timeunit_profit( | ||||||
|  |                 timescale, | ||||||
|  |                 stake_cur, | ||||||
|  |                 fiat_disp_cur, | ||||||
|  |                 unit | ||||||
|  |             ) | ||||||
|  |             stats_tab = tabulate( | ||||||
|  |                 [[f"{period['date']} ({period['trade_count']})", | ||||||
|  |                   f"{round_coin_value(period['abs_profit'], stats['stake_currency'])}", | ||||||
|  |                   f"{period['fiat_value']:.2f} {stats['fiat_display_currency']}", | ||||||
|  |                   f"{period['rel_profit']:.2%}", | ||||||
|  |                   ] for period in stats['data']], | ||||||
|  |                 headers=[ | ||||||
|  |                     f"{val.header} (count)", | ||||||
|  |                     f'{stake_cur}', | ||||||
|  |                     f'{fiat_disp_cur}', | ||||||
|  |                     'Profit %', | ||||||
|  |                     'Trades', | ||||||
|  |                 ], | ||||||
|  |                 tablefmt='simple') | ||||||
|  |             message = ( | ||||||
|  |                 f'<b>{val.message} Profit over the last {timescale} {val.message2}</b>:\n' | ||||||
|  |                 f'<pre>{stats_tab}</pre>' | ||||||
|  |             ) | ||||||
|  |             self._send_msg(message, parse_mode=ParseMode.HTML, reload_able=True, | ||||||
|  |                            callback_path=val.callback, query=update.callback_query) | ||||||
|  |         except RPCException as e: | ||||||
|  |             self._send_msg(str(e)) | ||||||
|  |  | ||||||
|     @authorized_only |     @authorized_only | ||||||
|     def _daily(self, update: Update, context: CallbackContext) -> None: |     def _daily(self, update: Update, context: CallbackContext) -> None: | ||||||
|         """ |         """ | ||||||
| @@ -580,35 +644,7 @@ class Telegram(RPCHandler): | |||||||
|         :param update: message update |         :param update: message update | ||||||
|         :return: None |         :return: None | ||||||
|         """ |         """ | ||||||
|         stake_cur = self._config['stake_currency'] |         self._timeunit_stats(update, context, 'days') | ||||||
|         fiat_disp_cur = self._config.get('fiat_display_currency', '') |  | ||||||
|         try: |  | ||||||
|             timescale = int(context.args[0]) if context.args else 7 |  | ||||||
|         except (TypeError, ValueError, IndexError): |  | ||||||
|             timescale = 7 |  | ||||||
|         try: |  | ||||||
|             stats = self._rpc._rpc_daily_profit( |  | ||||||
|                 timescale, |  | ||||||
|                 stake_cur, |  | ||||||
|                 fiat_disp_cur |  | ||||||
|             ) |  | ||||||
|             stats_tab = tabulate( |  | ||||||
|                 [[day['date'], |  | ||||||
|                   f"{round_coin_value(day['abs_profit'], stats['stake_currency'])}", |  | ||||||
|                   f"{day['fiat_value']:.3f} {stats['fiat_display_currency']}", |  | ||||||
|                   f"{day['trade_count']} trades"] for day in stats['data']], |  | ||||||
|                 headers=[ |  | ||||||
|                     'Day', |  | ||||||
|                     f'Profit {stake_cur}', |  | ||||||
|                     f'Profit {fiat_disp_cur}', |  | ||||||
|                     'Trades', |  | ||||||
|                 ], |  | ||||||
|                 tablefmt='simple') |  | ||||||
|             message = f'<b>Daily Profit over the last {timescale} days</b>:\n<pre>{stats_tab}</pre>' |  | ||||||
|             self._send_msg(message, parse_mode=ParseMode.HTML, reload_able=True, |  | ||||||
|                            callback_path="update_daily", query=update.callback_query) |  | ||||||
|         except RPCException as e: |  | ||||||
|             self._send_msg(str(e)) |  | ||||||
|  |  | ||||||
|     @authorized_only |     @authorized_only | ||||||
|     def _weekly(self, update: Update, context: CallbackContext) -> None: |     def _weekly(self, update: Update, context: CallbackContext) -> None: | ||||||
| @@ -619,36 +655,7 @@ class Telegram(RPCHandler): | |||||||
|         :param update: message update |         :param update: message update | ||||||
|         :return: None |         :return: None | ||||||
|         """ |         """ | ||||||
|         stake_cur = self._config['stake_currency'] |         self._timeunit_stats(update, context, 'weeks') | ||||||
|         fiat_disp_cur = self._config.get('fiat_display_currency', '') |  | ||||||
|         try: |  | ||||||
|             timescale = int(context.args[0]) if context.args else 8 |  | ||||||
|         except (TypeError, ValueError, IndexError): |  | ||||||
|             timescale = 8 |  | ||||||
|         try: |  | ||||||
|             stats = self._rpc._rpc_weekly_profit( |  | ||||||
|                 timescale, |  | ||||||
|                 stake_cur, |  | ||||||
|                 fiat_disp_cur |  | ||||||
|             ) |  | ||||||
|             stats_tab = tabulate( |  | ||||||
|                 [[week['date'], |  | ||||||
|                   f"{round_coin_value(week['abs_profit'], stats['stake_currency'])}", |  | ||||||
|                   f"{week['fiat_value']:.3f} {stats['fiat_display_currency']}", |  | ||||||
|                   f"{week['trade_count']} trades"] for week in stats['data']], |  | ||||||
|                 headers=[ |  | ||||||
|                     'Monday', |  | ||||||
|                     f'Profit {stake_cur}', |  | ||||||
|                     f'Profit {fiat_disp_cur}', |  | ||||||
|                     'Trades', |  | ||||||
|                 ], |  | ||||||
|                 tablefmt='simple') |  | ||||||
|             message = f'<b>Weekly Profit over the last {timescale} weeks ' \ |  | ||||||
|                       f'(starting from Monday)</b>:\n<pre>{stats_tab}</pre> ' |  | ||||||
|             self._send_msg(message, parse_mode=ParseMode.HTML, reload_able=True, |  | ||||||
|                            callback_path="update_weekly", query=update.callback_query) |  | ||||||
|         except RPCException as e: |  | ||||||
|             self._send_msg(str(e)) |  | ||||||
|  |  | ||||||
|     @authorized_only |     @authorized_only | ||||||
|     def _monthly(self, update: Update, context: CallbackContext) -> None: |     def _monthly(self, update: Update, context: CallbackContext) -> None: | ||||||
| @@ -659,36 +666,7 @@ class Telegram(RPCHandler): | |||||||
|         :param update: message update |         :param update: message update | ||||||
|         :return: None |         :return: None | ||||||
|         """ |         """ | ||||||
|         stake_cur = self._config['stake_currency'] |         self._timeunit_stats(update, context, 'months') | ||||||
|         fiat_disp_cur = self._config.get('fiat_display_currency', '') |  | ||||||
|         try: |  | ||||||
|             timescale = int(context.args[0]) if context.args else 6 |  | ||||||
|         except (TypeError, ValueError, IndexError): |  | ||||||
|             timescale = 6 |  | ||||||
|         try: |  | ||||||
|             stats = self._rpc._rpc_monthly_profit( |  | ||||||
|                 timescale, |  | ||||||
|                 stake_cur, |  | ||||||
|                 fiat_disp_cur |  | ||||||
|             ) |  | ||||||
|             stats_tab = tabulate( |  | ||||||
|                 [[month['date'], |  | ||||||
|                   f"{round_coin_value(month['abs_profit'], stats['stake_currency'])}", |  | ||||||
|                   f"{month['fiat_value']:.3f} {stats['fiat_display_currency']}", |  | ||||||
|                   f"{month['trade_count']} trades"] for month in stats['data']], |  | ||||||
|                 headers=[ |  | ||||||
|                     'Month', |  | ||||||
|                     f'Profit {stake_cur}', |  | ||||||
|                     f'Profit {fiat_disp_cur}', |  | ||||||
|                     'Trades', |  | ||||||
|                 ], |  | ||||||
|                 tablefmt='simple') |  | ||||||
|             message = f'<b>Monthly Profit over the last {timescale} months' \ |  | ||||||
|                       f'</b>:\n<pre>{stats_tab}</pre> ' |  | ||||||
|             self._send_msg(message, parse_mode=ParseMode.HTML, reload_able=True, |  | ||||||
|                            callback_path="update_monthly", query=update.callback_query) |  | ||||||
|         except RPCException as e: |  | ||||||
|             self._send_msg(str(e)) |  | ||||||
|  |  | ||||||
|     @authorized_only |     @authorized_only | ||||||
|     def _profit(self, update: Update, context: CallbackContext) -> None: |     def _profit(self, update: Update, context: CallbackContext) -> None: | ||||||
|   | |||||||
| @@ -289,6 +289,7 @@ class IStrategy(ABC, HyperStrategyMixin): | |||||||
|         :param order_type: Order type (as configured in order_types). usually limit or market. |         :param order_type: Order type (as configured in order_types). usually limit or market. | ||||||
|         :param amount: Amount in target (base) currency that's going to be traded. |         :param amount: Amount in target (base) currency that's going to be traded. | ||||||
|         :param rate: Rate that's going to be used when using limit orders |         :param rate: Rate that's going to be used when using limit orders | ||||||
|  |                      or current rate for market orders. | ||||||
|         :param time_in_force: Time in force. Defaults to GTC (Good-til-cancelled). |         :param time_in_force: Time in force. Defaults to GTC (Good-til-cancelled). | ||||||
|         :param current_time: datetime object, containing the current datetime |         :param current_time: datetime object, containing the current datetime | ||||||
|         :param entry_tag: Optional entry_tag (buy_tag) if provided with the buy signal. |         :param entry_tag: Optional entry_tag (buy_tag) if provided with the buy signal. | ||||||
| @@ -316,6 +317,7 @@ class IStrategy(ABC, HyperStrategyMixin): | |||||||
|         :param order_type: Order type (as configured in order_types). usually limit or market. |         :param order_type: Order type (as configured in order_types). usually limit or market. | ||||||
|         :param amount: Amount in base currency. |         :param amount: Amount in base currency. | ||||||
|         :param rate: Rate that's going to be used when using limit orders |         :param rate: Rate that's going to be used when using limit orders | ||||||
|  |                      or current rate for market orders. | ||||||
|         :param time_in_force: Time in force. Defaults to GTC (Good-til-cancelled). |         :param time_in_force: Time in force. Defaults to GTC (Good-til-cancelled). | ||||||
|         :param exit_reason: Exit reason. |         :param exit_reason: Exit reason. | ||||||
|             Can be any of ['roi', 'stop_loss', 'stoploss_on_exchange', 'trailing_stop_loss', |             Can be any of ['roi', 'stop_loss', 'stoploss_on_exchange', 'trailing_stop_loss', | ||||||
| @@ -509,8 +511,8 @@ class IStrategy(ABC, HyperStrategyMixin): | |||||||
|         return current_order_rate |         return current_order_rate | ||||||
|  |  | ||||||
|     def leverage(self, pair: str, current_time: datetime, current_rate: float, |     def leverage(self, pair: str, current_time: datetime, current_rate: float, | ||||||
|                  proposed_leverage: float, max_leverage: float, side: str, |                  proposed_leverage: float, max_leverage: float, entry_tag: Optional[str], | ||||||
|                  **kwargs) -> float: |                  side: str, **kwargs) -> float: | ||||||
|         """ |         """ | ||||||
|         Customize leverage for each new trade. This method is only called in futures mode. |         Customize leverage for each new trade. This method is only called in futures mode. | ||||||
|  |  | ||||||
| @@ -519,6 +521,7 @@ class IStrategy(ABC, HyperStrategyMixin): | |||||||
|         :param current_rate: Rate, calculated based on pricing settings in exit_pricing. |         :param current_rate: Rate, calculated based on pricing settings in exit_pricing. | ||||||
|         :param proposed_leverage: A leverage proposed by the bot. |         :param proposed_leverage: A leverage proposed by the bot. | ||||||
|         :param max_leverage: Max leverage allowed on this pair |         :param max_leverage: Max leverage allowed on this pair | ||||||
|  |         :param entry_tag: Optional entry_tag (buy_tag) if provided with the buy signal. | ||||||
|         :param side: 'long' or 'short' - indicating the direction of the proposed trade |         :param side: 'long' or 'short' - indicating the direction of the proposed trade | ||||||
|         :return: A leverage amount, which is between 1.0 and max_leverage. |         :return: A leverage amount, which is between 1.0 and max_leverage. | ||||||
|         """ |         """ | ||||||
|   | |||||||
| @@ -161,6 +161,7 @@ def confirm_trade_entry(self, pair: str, order_type: str, amount: float, rate: f | |||||||
|     :param order_type: Order type (as configured in order_types). usually limit or market. |     :param order_type: Order type (as configured in order_types). usually limit or market. | ||||||
|     :param amount: Amount in target (base) currency that's going to be traded. |     :param amount: Amount in target (base) currency that's going to be traded. | ||||||
|     :param rate: Rate that's going to be used when using limit orders |     :param rate: Rate that's going to be used when using limit orders | ||||||
|  |                  or current rate for market orders. | ||||||
|     :param time_in_force: Time in force. Defaults to GTC (Good-til-cancelled). |     :param time_in_force: Time in force. Defaults to GTC (Good-til-cancelled). | ||||||
|     :param current_time: datetime object, containing the current datetime |     :param current_time: datetime object, containing the current datetime | ||||||
|     :param entry_tag: Optional entry_tag (buy_tag) if provided with the buy signal. |     :param entry_tag: Optional entry_tag (buy_tag) if provided with the buy signal. | ||||||
| @@ -188,6 +189,7 @@ def confirm_trade_exit(self, pair: str, trade: 'Trade', order_type: str, amount: | |||||||
|     :param order_type: Order type (as configured in order_types). usually limit or market. |     :param order_type: Order type (as configured in order_types). usually limit or market. | ||||||
|     :param amount: Amount in base currency. |     :param amount: Amount in base currency. | ||||||
|     :param rate: Rate that's going to be used when using limit orders |     :param rate: Rate that's going to be used when using limit orders | ||||||
|  |                  or current rate for market orders. | ||||||
|     :param time_in_force: Time in force. Defaults to GTC (Good-til-cancelled). |     :param time_in_force: Time in force. Defaults to GTC (Good-til-cancelled). | ||||||
|     :param exit_reason: Exit reason. |     :param exit_reason: Exit reason. | ||||||
|         Can be any of ['roi', 'stop_loss', 'stoploss_on_exchange', 'trailing_stop_loss', |         Can be any of ['roi', 'stop_loss', 'stoploss_on_exchange', 'trailing_stop_loss', | ||||||
| @@ -267,8 +269,8 @@ def adjust_trade_position(self, trade: 'Trade', current_time: 'datetime', | |||||||
|     return None |     return None | ||||||
|  |  | ||||||
| def leverage(self, pair: str, current_time: datetime, current_rate: float, | def leverage(self, pair: str, current_time: datetime, current_rate: float, | ||||||
|                 proposed_leverage: float, max_leverage: float, side: str, |              proposed_leverage: float, max_leverage: float, entry_tag: Optional[str], | ||||||
|                 **kwargs) -> float: |              side: str, **kwargs) -> float: | ||||||
|     """ |     """ | ||||||
|     Customize leverage for each new trade. This method is only called in futures mode. |     Customize leverage for each new trade. This method is only called in futures mode. | ||||||
|  |  | ||||||
| @@ -277,6 +279,7 @@ def leverage(self, pair: str, current_time: datetime, current_rate: float, | |||||||
|     :param current_rate: Rate, calculated based on pricing settings in exit_pricing. |     :param current_rate: Rate, calculated based on pricing settings in exit_pricing. | ||||||
|     :param proposed_leverage: A leverage proposed by the bot. |     :param proposed_leverage: A leverage proposed by the bot. | ||||||
|     :param max_leverage: Max leverage allowed on this pair |     :param max_leverage: Max leverage allowed on this pair | ||||||
|  |     :param entry_tag: Optional entry_tag (buy_tag) if provided with the buy signal. | ||||||
|     :param side: 'long' or 'short' - indicating the direction of the proposed trade |     :param side: 'long' or 'short' - indicating the direction of the proposed trade | ||||||
|     :return: A leverage amount, which is between 1.0 and max_leverage. |     :return: A leverage amount, which is between 1.0 and max_leverage. | ||||||
|     """ |     """ | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ | |||||||
| coveralls==3.3.1 | coveralls==3.3.1 | ||||||
| flake8==4.0.1 | flake8==4.0.1 | ||||||
| flake8-tidy-imports==4.8.0 | flake8-tidy-imports==4.8.0 | ||||||
| mypy==0.960 | mypy==0.961 | ||||||
| pre-commit==2.19.0 | pre-commit==2.19.0 | ||||||
| pytest==7.1.2 | pytest==7.1.2 | ||||||
| pytest-asyncio==0.18.3 | pytest-asyncio==0.18.3 | ||||||
| @@ -23,7 +23,7 @@ nbconvert==6.5.0 | |||||||
|  |  | ||||||
| # mypy types | # mypy types | ||||||
| types-cachetools==5.0.1 | types-cachetools==5.0.1 | ||||||
| types-filelock==3.2.6 | types-filelock==3.2.7 | ||||||
| types-requests==2.27.29 | types-requests==2.27.30 | ||||||
| types-tabulate==0.8.9 | types-tabulate==0.8.9 | ||||||
| types-python-dateutil==2.8.17 | types-python-dateutil==2.8.17 | ||||||
|   | |||||||
| @@ -5,5 +5,5 @@ | |||||||
| scipy==1.8.1 | scipy==1.8.1 | ||||||
| scikit-learn==1.1.1 | scikit-learn==1.1.1 | ||||||
| scikit-optimize==0.9.0 | scikit-optimize==0.9.0 | ||||||
| filelock==3.7.0 | filelock==3.7.1 | ||||||
| progressbar2==4.0.0 | progressbar2==4.0.0 | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Include all requirements to run the bot. | # Include all requirements to run the bot. | ||||||
| -r requirements.txt | -r requirements.txt | ||||||
|  |  | ||||||
| plotly==5.8.0 | plotly==5.8.2 | ||||||
|   | |||||||
| @@ -2,17 +2,17 @@ numpy==1.22.4 | |||||||
| pandas==1.4.2 | pandas==1.4.2 | ||||||
| pandas-ta==0.3.14b | pandas-ta==0.3.14b | ||||||
|  |  | ||||||
| ccxt==1.84.39 | ccxt==1.87.12 | ||||||
| # Pin cryptography for now due to rust build errors with piwheels | # Pin cryptography for now due to rust build errors with piwheels | ||||||
| cryptography==37.0.2 | cryptography==37.0.2 | ||||||
| aiohttp==3.8.1 | aiohttp==3.8.1 | ||||||
| SQLAlchemy==1.4.36 | SQLAlchemy==1.4.37 | ||||||
| python-telegram-bot==13.12 | python-telegram-bot==13.12 | ||||||
| arrow==1.2.2 | arrow==1.2.2 | ||||||
| cachetools==4.2.2 | cachetools==4.2.2 | ||||||
| requests==2.27.1 | requests==2.28.0 | ||||||
| urllib3==1.26.9 | urllib3==1.26.9 | ||||||
| jsonschema==4.5.1 | jsonschema==4.6.0 | ||||||
| TA-Lib==0.4.24 | TA-Lib==0.4.24 | ||||||
| technical==1.3.0 | technical==1.3.0 | ||||||
| tabulate==0.8.9 | tabulate==0.8.9 | ||||||
| @@ -28,7 +28,7 @@ py_find_1st==1.1.5 | |||||||
| # Load ticker files 30% faster | # Load ticker files 30% faster | ||||||
| python-rapidjson==1.6 | python-rapidjson==1.6 | ||||||
| # Properly format api responses | # Properly format api responses | ||||||
| orjson==3.6.8 | orjson==3.7.2 | ||||||
|  |  | ||||||
| # Notify systemd | # Notify systemd | ||||||
| sdnotify==0.3.2 | sdnotify==0.3.2 | ||||||
|   | |||||||
| @@ -261,7 +261,7 @@ class FtRestClient(): | |||||||
|                 } |                 } | ||||||
|         return self._post("forcebuy", data=data) |         return self._post("forcebuy", data=data) | ||||||
|  |  | ||||||
|     def force_enter(self, pair, side, price=None): |     def forceenter(self, pair, side, price=None): | ||||||
|         """Force entering a trade |         """Force entering a trade | ||||||
|  |  | ||||||
|         :param pair: Pair to buy (ETH/BTC) |         :param pair: Pair to buy (ETH/BTC) | ||||||
| @@ -273,7 +273,7 @@ class FtRestClient(): | |||||||
|                 "side": side, |                 "side": side, | ||||||
|                 "price": price, |                 "price": price, | ||||||
|                 } |                 } | ||||||
|         return self._post("force_enter", data=data) |         return self._post("forceenter", data=data) | ||||||
|  |  | ||||||
|     def forceexit(self, tradeid): |     def forceexit(self, tradeid): | ||||||
|         """Force-exit a trade. |         """Force-exit a trade. | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								setup.sh
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								setup.sh
									
									
									
									
									
								
							| @@ -87,6 +87,10 @@ function updateenv() { | |||||||
|         echo "Failed installing Freqtrade" |         echo "Failed installing Freqtrade" | ||||||
|         exit 1 |         exit 1 | ||||||
|     fi |     fi | ||||||
|  |  | ||||||
|  |     echo "Installing freqUI" | ||||||
|  |     freqtrade install-ui | ||||||
|  |  | ||||||
|     echo "pip install completed" |     echo "pip install completed" | ||||||
|     echo |     echo | ||||||
|     if [[ $dev =~ ^[Yy]$ ]]; then |     if [[ $dev =~ ^[Yy]$ ]]; then | ||||||
|   | |||||||
| @@ -325,7 +325,7 @@ def create_mock_trades_with_leverage(fee, use_db: bool = True): | |||||||
|         Trade.query.session.flush() |         Trade.query.session.flush() | ||||||
|  |  | ||||||
|  |  | ||||||
| def create_mock_trades_usdt(fee, use_db: bool = True): | def create_mock_trades_usdt(fee, is_short: Optional[bool] = False, use_db: bool = True): | ||||||
|     """ |     """ | ||||||
|     Create some fake trades ... |     Create some fake trades ... | ||||||
|     """ |     """ | ||||||
| @@ -335,26 +335,29 @@ def create_mock_trades_usdt(fee, use_db: bool = True): | |||||||
|         else: |         else: | ||||||
|             LocalTrade.add_bt_trade(trade) |             LocalTrade.add_bt_trade(trade) | ||||||
|  |  | ||||||
|  |     is_short1 = is_short if is_short is not None else True | ||||||
|  |     is_short2 = is_short if is_short is not None else False | ||||||
|  |  | ||||||
|     # Simulate dry_run entries |     # Simulate dry_run entries | ||||||
|     trade = mock_trade_usdt_1(fee) |     trade = mock_trade_usdt_1(fee, is_short1) | ||||||
|     add_trade(trade) |     add_trade(trade) | ||||||
|  |  | ||||||
|     trade = mock_trade_usdt_2(fee) |     trade = mock_trade_usdt_2(fee, is_short1) | ||||||
|     add_trade(trade) |     add_trade(trade) | ||||||
|  |  | ||||||
|     trade = mock_trade_usdt_3(fee) |     trade = mock_trade_usdt_3(fee, is_short1) | ||||||
|     add_trade(trade) |     add_trade(trade) | ||||||
|  |  | ||||||
|     trade = mock_trade_usdt_4(fee) |     trade = mock_trade_usdt_4(fee, is_short2) | ||||||
|     add_trade(trade) |     add_trade(trade) | ||||||
|  |  | ||||||
|     trade = mock_trade_usdt_5(fee) |     trade = mock_trade_usdt_5(fee, is_short2) | ||||||
|     add_trade(trade) |     add_trade(trade) | ||||||
|  |  | ||||||
|     trade = mock_trade_usdt_6(fee) |     trade = mock_trade_usdt_6(fee, is_short1) | ||||||
|     add_trade(trade) |     add_trade(trade) | ||||||
|  |  | ||||||
|     trade = mock_trade_usdt_7(fee) |     trade = mock_trade_usdt_7(fee, is_short1) | ||||||
|     add_trade(trade) |     add_trade(trade) | ||||||
|     if use_db: |     if use_db: | ||||||
|         Trade.commit() |         Trade.commit() | ||||||
|   | |||||||
| @@ -6,47 +6,84 @@ from freqtrade.persistence.models import Order, Trade | |||||||
| MOCK_TRADE_COUNT = 6 | MOCK_TRADE_COUNT = 6 | ||||||
|  |  | ||||||
|  |  | ||||||
| def mock_order_usdt_1(): | def entry_side(is_short: bool): | ||||||
|  |     return "sell" if is_short else "buy" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def exit_side(is_short: bool): | ||||||
|  |     return "buy" if is_short else "sell" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def direc(is_short: bool): | ||||||
|  |     return "short" if is_short else "long" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def mock_order_usdt_1(is_short: bool): | ||||||
|     return { |     return { | ||||||
|         'id': '1234', |         'id': f'prod_entry_1_{direc(is_short)}', | ||||||
|         'symbol': 'ADA/USDT', |         'symbol': 'LTC/USDT', | ||||||
|         'status': 'closed', |         'status': 'closed', | ||||||
|         'side': 'buy', |         'side': entry_side(is_short), | ||||||
|         'type': 'limit', |         'type': 'limit', | ||||||
|         'price': 2.0, |         'price': 10.0, | ||||||
|         'amount': 10.0, |         'amount': 2.0, | ||||||
|         'filled': 10.0, |         'filled': 2.0, | ||||||
|         'remaining': 0.0, |         'remaining': 0.0, | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
| def mock_trade_usdt_1(fee): | def mock_order_usdt_1_exit(is_short: bool): | ||||||
|  |     return { | ||||||
|  |         'id': f'prod_exit_1_{direc(is_short)}', | ||||||
|  |         'symbol': 'LTC/USDT', | ||||||
|  |         'status': 'closed', | ||||||
|  |         'side': exit_side(is_short), | ||||||
|  |         'type': 'limit', | ||||||
|  |         'price': 8.0, | ||||||
|  |         'amount': 2.0, | ||||||
|  |         'filled': 2.0, | ||||||
|  |         'remaining': 0.0, | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def mock_trade_usdt_1(fee, is_short: bool): | ||||||
|  |     """ | ||||||
|  |     Simulate prod entry with open sell order | ||||||
|  |     """ | ||||||
|     trade = Trade( |     trade = Trade( | ||||||
|         pair='ADA/USDT', |         pair='LTC/USDT', | ||||||
|         stake_amount=20.0, |         stake_amount=20.0, | ||||||
|         amount=10.0, |         amount=2.0, | ||||||
|         amount_requested=10.0, |         amount_requested=2.0, | ||||||
|  |         open_date=datetime.now(tz=timezone.utc) - timedelta(days=2, minutes=20), | ||||||
|  |         close_date=datetime.now(tz=timezone.utc) - timedelta(days=2, minutes=5), | ||||||
|         fee_open=fee.return_value, |         fee_open=fee.return_value, | ||||||
|         fee_close=fee.return_value, |         fee_close=fee.return_value, | ||||||
|         is_open=True, |         is_open=False, | ||||||
|         open_date=datetime.now(tz=timezone.utc) - timedelta(minutes=17), |         open_rate=10.0, | ||||||
|         open_rate=2.0, |         close_rate=8.0, | ||||||
|  |         close_profit=-0.2, | ||||||
|  |         close_profit_abs=-4.0, | ||||||
|         exchange='binance', |         exchange='binance', | ||||||
|         open_order_id='dry_run_buy_12345', |         strategy='SampleStrategy', | ||||||
|         strategy='StrategyTestV2', |         open_order_id=f'prod_exit_1_{direc(is_short)}', | ||||||
|         timeframe=5, |         timeframe=5, | ||||||
|  |         is_short=is_short, | ||||||
|     ) |     ) | ||||||
|     o = Order.parse_from_ccxt_object(mock_order_usdt_1(), 'ADA/USDT', 'buy') |     o = Order.parse_from_ccxt_object(mock_order_usdt_1(is_short), 'LTC/USDT', entry_side(is_short)) | ||||||
|  |     trade.orders.append(o) | ||||||
|  |     o = Order.parse_from_ccxt_object(mock_order_usdt_1_exit(is_short), | ||||||
|  |                                      'LTC/USDT', exit_side(is_short)) | ||||||
|     trade.orders.append(o) |     trade.orders.append(o) | ||||||
|     return trade |     return trade | ||||||
|  |  | ||||||
|  |  | ||||||
| def mock_order_usdt_2(): | def mock_order_usdt_2(is_short: bool): | ||||||
|     return { |     return { | ||||||
|         'id': '1235', |         'id': f'1235_{direc(is_short)}', | ||||||
|         'symbol': 'ETC/USDT', |         'symbol': 'ETC/USDT', | ||||||
|         'status': 'closed', |         'status': 'closed', | ||||||
|         'side': 'buy', |         'side': entry_side(is_short), | ||||||
|         'type': 'limit', |         'type': 'limit', | ||||||
|         'price': 2.0, |         'price': 2.0, | ||||||
|         'amount': 100.0, |         'amount': 100.0, | ||||||
| @@ -55,12 +92,12 @@ def mock_order_usdt_2(): | |||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
| def mock_order_usdt_2_sell(): | def mock_order_usdt_2_exit(is_short: bool): | ||||||
|     return { |     return { | ||||||
|         'id': '12366', |         'id': f'12366_{direc(is_short)}', | ||||||
|         'symbol': 'ETC/USDT', |         'symbol': 'ETC/USDT', | ||||||
|         'status': 'closed', |         'status': 'closed', | ||||||
|         'side': 'sell', |         'side': exit_side(is_short), | ||||||
|         'type': 'limit', |         'type': 'limit', | ||||||
|         'price': 2.05, |         'price': 2.05, | ||||||
|         'amount': 100.0, |         'amount': 100.0, | ||||||
| @@ -69,7 +106,7 @@ def mock_order_usdt_2_sell(): | |||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
| def mock_trade_usdt_2(fee): | def mock_trade_usdt_2(fee, is_short: bool): | ||||||
|     """ |     """ | ||||||
|     Closed trade... |     Closed trade... | ||||||
|     """ |     """ | ||||||
| @@ -82,30 +119,33 @@ def mock_trade_usdt_2(fee): | |||||||
|         fee_close=fee.return_value, |         fee_close=fee.return_value, | ||||||
|         open_rate=2.0, |         open_rate=2.0, | ||||||
|         close_rate=2.05, |         close_rate=2.05, | ||||||
|         close_profit=5.0, |         close_profit=0.05, | ||||||
|         close_profit_abs=3.9875, |         close_profit_abs=3.9875, | ||||||
|         exchange='binance', |         exchange='binance', | ||||||
|         is_open=False, |         is_open=False, | ||||||
|         open_order_id='dry_run_sell_12345', |         open_order_id=f'12366_{direc(is_short)}', | ||||||
|         strategy='StrategyTestV2', |         strategy='StrategyTestV2', | ||||||
|         timeframe=5, |         timeframe=5, | ||||||
|         exit_reason='sell_signal', |         enter_tag='TEST1', | ||||||
|  |         exit_reason='exit_signal', | ||||||
|         open_date=datetime.now(tz=timezone.utc) - timedelta(minutes=20), |         open_date=datetime.now(tz=timezone.utc) - timedelta(minutes=20), | ||||||
|         close_date=datetime.now(tz=timezone.utc) - timedelta(minutes=2), |         close_date=datetime.now(tz=timezone.utc) - timedelta(minutes=2), | ||||||
|  |         is_short=is_short, | ||||||
|     ) |     ) | ||||||
|     o = Order.parse_from_ccxt_object(mock_order_usdt_2(), 'ETC/USDT', 'buy') |     o = Order.parse_from_ccxt_object(mock_order_usdt_2(is_short), 'ETC/USDT', entry_side(is_short)) | ||||||
|     trade.orders.append(o) |     trade.orders.append(o) | ||||||
|     o = Order.parse_from_ccxt_object(mock_order_usdt_2_sell(), 'ETC/USDT', 'sell') |     o = Order.parse_from_ccxt_object( | ||||||
|  |         mock_order_usdt_2_exit(is_short), 'ETC/USDT', exit_side(is_short)) | ||||||
|     trade.orders.append(o) |     trade.orders.append(o) | ||||||
|     return trade |     return trade | ||||||
|  |  | ||||||
|  |  | ||||||
| def mock_order_usdt_3(): | def mock_order_usdt_3(is_short: bool): | ||||||
|     return { |     return { | ||||||
|         'id': '41231a12a', |         'id': f'41231a12a_{direc(is_short)}', | ||||||
|         'symbol': 'XRP/USDT', |         'symbol': 'XRP/USDT', | ||||||
|         'status': 'closed', |         'status': 'closed', | ||||||
|         'side': 'buy', |         'side': entry_side(is_short), | ||||||
|         'type': 'limit', |         'type': 'limit', | ||||||
|         'price': 1.0, |         'price': 1.0, | ||||||
|         'amount': 30.0, |         'amount': 30.0, | ||||||
| @@ -114,12 +154,12 @@ def mock_order_usdt_3(): | |||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
| def mock_order_usdt_3_sell(): | def mock_order_usdt_3_exit(is_short: bool): | ||||||
|     return { |     return { | ||||||
|         'id': '41231a666a', |         'id': f'41231a666a_{direc(is_short)}', | ||||||
|         'symbol': 'XRP/USDT', |         'symbol': 'XRP/USDT', | ||||||
|         'status': 'closed', |         'status': 'closed', | ||||||
|         'side': 'sell', |         'side': exit_side(is_short), | ||||||
|         'type': 'stop_loss_limit', |         'type': 'stop_loss_limit', | ||||||
|         'price': 1.1, |         'price': 1.1, | ||||||
|         'average': 1.1, |         'average': 1.1, | ||||||
| @@ -129,7 +169,7 @@ def mock_order_usdt_3_sell(): | |||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
| def mock_trade_usdt_3(fee): | def mock_trade_usdt_3(fee, is_short: bool): | ||||||
|     """ |     """ | ||||||
|     Closed trade |     Closed trade | ||||||
|     """ |     """ | ||||||
| @@ -142,29 +182,32 @@ def mock_trade_usdt_3(fee): | |||||||
|         fee_close=fee.return_value, |         fee_close=fee.return_value, | ||||||
|         open_rate=1.0, |         open_rate=1.0, | ||||||
|         close_rate=1.1, |         close_rate=1.1, | ||||||
|         close_profit=10.0, |         close_profit=0.1, | ||||||
|         close_profit_abs=9.8425, |         close_profit_abs=9.8425, | ||||||
|         exchange='binance', |         exchange='binance', | ||||||
|         is_open=False, |         is_open=False, | ||||||
|         strategy='StrategyTestV2', |         strategy='StrategyTestV2', | ||||||
|         timeframe=5, |         timeframe=5, | ||||||
|  |         enter_tag='TEST3', | ||||||
|         exit_reason='roi', |         exit_reason='roi', | ||||||
|         open_date=datetime.now(tz=timezone.utc) - timedelta(minutes=20), |         open_date=datetime.now(tz=timezone.utc) - timedelta(minutes=20), | ||||||
|         close_date=datetime.now(tz=timezone.utc), |         close_date=datetime.now(tz=timezone.utc), | ||||||
|  |         is_short=is_short, | ||||||
|     ) |     ) | ||||||
|     o = Order.parse_from_ccxt_object(mock_order_usdt_3(), 'XRP/USDT', 'buy') |     o = Order.parse_from_ccxt_object(mock_order_usdt_3(is_short), 'XRP/USDT', entry_side(is_short)) | ||||||
|     trade.orders.append(o) |     trade.orders.append(o) | ||||||
|     o = Order.parse_from_ccxt_object(mock_order_usdt_3_sell(), 'XRP/USDT', 'sell') |     o = Order.parse_from_ccxt_object(mock_order_usdt_3_exit(is_short), | ||||||
|  |                                      'XRP/USDT', exit_side(is_short)) | ||||||
|     trade.orders.append(o) |     trade.orders.append(o) | ||||||
|     return trade |     return trade | ||||||
|  |  | ||||||
|  |  | ||||||
| def mock_order_usdt_4(): | def mock_order_usdt_4(is_short: bool): | ||||||
|     return { |     return { | ||||||
|         'id': 'prod_buy_12345', |         'id': f'prod_buy_12345_{direc(is_short)}', | ||||||
|         'symbol': 'ETC/USDT', |         'symbol': 'ETC/USDT', | ||||||
|         'status': 'open', |         'status': 'open', | ||||||
|         'side': 'buy', |         'side': entry_side(is_short), | ||||||
|         'type': 'limit', |         'type': 'limit', | ||||||
|         'price': 2.0, |         'price': 2.0, | ||||||
|         'amount': 10.0, |         'amount': 10.0, | ||||||
| @@ -173,7 +216,7 @@ def mock_order_usdt_4(): | |||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
| def mock_trade_usdt_4(fee): | def mock_trade_usdt_4(fee, is_short: bool): | ||||||
|     """ |     """ | ||||||
|     Simulate prod entry |     Simulate prod entry | ||||||
|     """ |     """ | ||||||
| @@ -188,21 +231,22 @@ def mock_trade_usdt_4(fee): | |||||||
|         is_open=True, |         is_open=True, | ||||||
|         open_rate=2.0, |         open_rate=2.0, | ||||||
|         exchange='binance', |         exchange='binance', | ||||||
|         open_order_id='prod_buy_12345', |         open_order_id=f'prod_buy_12345_{direc(is_short)}', | ||||||
|         strategy='StrategyTestV2', |         strategy='StrategyTestV2', | ||||||
|         timeframe=5, |         timeframe=5, | ||||||
|  |         is_short=is_short, | ||||||
|     ) |     ) | ||||||
|     o = Order.parse_from_ccxt_object(mock_order_usdt_4(), 'ETC/USDT', 'buy') |     o = Order.parse_from_ccxt_object(mock_order_usdt_4(is_short), 'ETC/USDT', entry_side(is_short)) | ||||||
|     trade.orders.append(o) |     trade.orders.append(o) | ||||||
|     return trade |     return trade | ||||||
|  |  | ||||||
|  |  | ||||||
| def mock_order_usdt_5(): | def mock_order_usdt_5(is_short: bool): | ||||||
|     return { |     return { | ||||||
|         'id': 'prod_buy_3455', |         'id': f'prod_buy_3455_{direc(is_short)}', | ||||||
|         'symbol': 'XRP/USDT', |         'symbol': 'XRP/USDT', | ||||||
|         'status': 'closed', |         'status': 'closed', | ||||||
|         'side': 'buy', |         'side': entry_side(is_short), | ||||||
|         'type': 'limit', |         'type': 'limit', | ||||||
|         'price': 2.0, |         'price': 2.0, | ||||||
|         'amount': 10.0, |         'amount': 10.0, | ||||||
| @@ -211,12 +255,12 @@ def mock_order_usdt_5(): | |||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
| def mock_order_usdt_5_stoploss(): | def mock_order_usdt_5_stoploss(is_short: bool): | ||||||
|     return { |     return { | ||||||
|         'id': 'prod_stoploss_3455', |         'id': f'prod_stoploss_3455_{direc(is_short)}', | ||||||
|         'symbol': 'XRP/USDT', |         'symbol': 'XRP/USDT', | ||||||
|         'status': 'open', |         'status': 'open', | ||||||
|         'side': 'sell', |         'side': exit_side(is_short), | ||||||
|         'type': 'stop_loss_limit', |         'type': 'stop_loss_limit', | ||||||
|         'price': 2.0, |         'price': 2.0, | ||||||
|         'amount': 10.0, |         'amount': 10.0, | ||||||
| @@ -225,7 +269,7 @@ def mock_order_usdt_5_stoploss(): | |||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
| def mock_trade_usdt_5(fee): | def mock_trade_usdt_5(fee, is_short: bool): | ||||||
|     """ |     """ | ||||||
|     Simulate prod entry with stoploss |     Simulate prod entry with stoploss | ||||||
|     """ |     """ | ||||||
| @@ -241,22 +285,23 @@ def mock_trade_usdt_5(fee): | |||||||
|         open_rate=2.0, |         open_rate=2.0, | ||||||
|         exchange='binance', |         exchange='binance', | ||||||
|         strategy='SampleStrategy', |         strategy='SampleStrategy', | ||||||
|         stoploss_order_id='prod_stoploss_3455', |         stoploss_order_id=f'prod_stoploss_3455_{direc(is_short)}', | ||||||
|         timeframe=5, |         timeframe=5, | ||||||
|  |         is_short=is_short, | ||||||
|     ) |     ) | ||||||
|     o = Order.parse_from_ccxt_object(mock_order_usdt_5(), 'XRP/USDT', 'buy') |     o = Order.parse_from_ccxt_object(mock_order_usdt_5(is_short), 'XRP/USDT', entry_side(is_short)) | ||||||
|     trade.orders.append(o) |     trade.orders.append(o) | ||||||
|     o = Order.parse_from_ccxt_object(mock_order_usdt_5_stoploss(), 'XRP/USDT', 'stoploss') |     o = Order.parse_from_ccxt_object(mock_order_usdt_5_stoploss(is_short), 'XRP/USDT', 'stoploss') | ||||||
|     trade.orders.append(o) |     trade.orders.append(o) | ||||||
|     return trade |     return trade | ||||||
|  |  | ||||||
|  |  | ||||||
| def mock_order_usdt_6(): | def mock_order_usdt_6(is_short: bool): | ||||||
|     return { |     return { | ||||||
|         'id': 'prod_buy_6', |         'id': f'prod_entry_6_{direc(is_short)}', | ||||||
|         'symbol': 'LTC/USDT', |         'symbol': 'LTC/USDT', | ||||||
|         'status': 'closed', |         'status': 'closed', | ||||||
|         'side': 'buy', |         'side': entry_side(is_short), | ||||||
|         'type': 'limit', |         'type': 'limit', | ||||||
|         'price': 10.0, |         'price': 10.0, | ||||||
|         'amount': 2.0, |         'amount': 2.0, | ||||||
| @@ -265,12 +310,12 @@ def mock_order_usdt_6(): | |||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
| def mock_order_usdt_6_sell(): | def mock_order_usdt_6_exit(is_short: bool): | ||||||
|     return { |     return { | ||||||
|         'id': 'prod_sell_6', |         'id': f'prod_exit_6_{direc(is_short)}', | ||||||
|         'symbol': 'LTC/USDT', |         'symbol': 'LTC/USDT', | ||||||
|         'status': 'open', |         'status': 'open', | ||||||
|         'side': 'sell', |         'side': exit_side(is_short), | ||||||
|         'type': 'limit', |         'type': 'limit', | ||||||
|         'price': 12.0, |         'price': 12.0, | ||||||
|         'amount': 2.0, |         'amount': 2.0, | ||||||
| @@ -279,7 +324,7 @@ def mock_order_usdt_6_sell(): | |||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
| def mock_trade_usdt_6(fee): | def mock_trade_usdt_6(fee, is_short: bool): | ||||||
|     """ |     """ | ||||||
|     Simulate prod entry with open sell order |     Simulate prod entry with open sell order | ||||||
|     """ |     """ | ||||||
| @@ -295,69 +340,49 @@ def mock_trade_usdt_6(fee): | |||||||
|         open_rate=10.0, |         open_rate=10.0, | ||||||
|         exchange='binance', |         exchange='binance', | ||||||
|         strategy='SampleStrategy', |         strategy='SampleStrategy', | ||||||
|         open_order_id="prod_sell_6", |         open_order_id=f'prod_exit_6_{direc(is_short)}', | ||||||
|         timeframe=5, |         timeframe=5, | ||||||
|  |         is_short=is_short, | ||||||
|     ) |     ) | ||||||
|     o = Order.parse_from_ccxt_object(mock_order_usdt_6(), 'LTC/USDT', 'buy') |     o = Order.parse_from_ccxt_object(mock_order_usdt_6(is_short), 'LTC/USDT', entry_side(is_short)) | ||||||
|     trade.orders.append(o) |     trade.orders.append(o) | ||||||
|     o = Order.parse_from_ccxt_object(mock_order_usdt_6_sell(), 'LTC/USDT', 'sell') |     o = Order.parse_from_ccxt_object(mock_order_usdt_6_exit(is_short), | ||||||
|  |                                      'LTC/USDT', exit_side(is_short)) | ||||||
|     trade.orders.append(o) |     trade.orders.append(o) | ||||||
|     return trade |     return trade | ||||||
|  |  | ||||||
|  |  | ||||||
| def mock_order_usdt_7(): | def mock_order_usdt_7(is_short: bool): | ||||||
|     return { |     return { | ||||||
|         'id': 'prod_buy_7', |         'id': f'1234_{direc(is_short)}', | ||||||
|         'symbol': 'LTC/USDT', |         'symbol': 'ADA/USDT', | ||||||
|         'status': 'closed', |         'status': 'closed', | ||||||
|         'side': 'buy', |         'side': entry_side(is_short), | ||||||
|         'type': 'limit', |         'type': 'limit', | ||||||
|         'price': 10.0, |         'price': 2.0, | ||||||
|         'amount': 2.0, |         'amount': 10.0, | ||||||
|         'filled': 2.0, |         'filled': 10.0, | ||||||
|         'remaining': 0.0, |         'remaining': 0.0, | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
| def mock_order_usdt_7_sell(): | def mock_trade_usdt_7(fee, is_short: bool): | ||||||
|     return { |  | ||||||
|         'id': 'prod_sell_7', |  | ||||||
|         'symbol': 'LTC/USDT', |  | ||||||
|         'status': 'closed', |  | ||||||
|         'side': 'sell', |  | ||||||
|         'type': 'limit', |  | ||||||
|         'price': 8.0, |  | ||||||
|         'amount': 2.0, |  | ||||||
|         'filled': 2.0, |  | ||||||
|         'remaining': 0.0, |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def mock_trade_usdt_7(fee): |  | ||||||
|     """ |  | ||||||
|     Simulate prod entry with open sell order |  | ||||||
|     """ |  | ||||||
|     trade = Trade( |     trade = Trade( | ||||||
|         pair='LTC/USDT', |         pair='ADA/USDT', | ||||||
|         stake_amount=20.0, |         stake_amount=20.0, | ||||||
|         amount=2.0, |         amount=10.0, | ||||||
|         amount_requested=2.0, |         amount_requested=10.0, | ||||||
|         open_date=datetime.now(tz=timezone.utc) - timedelta(minutes=20), |  | ||||||
|         close_date=datetime.now(tz=timezone.utc) - timedelta(minutes=5), |  | ||||||
|         fee_open=fee.return_value, |         fee_open=fee.return_value, | ||||||
|         fee_close=fee.return_value, |         fee_close=fee.return_value, | ||||||
|         is_open=False, |         is_open=True, | ||||||
|         open_rate=10.0, |         open_date=datetime.now(tz=timezone.utc) - timedelta(minutes=17), | ||||||
|         close_rate=8.0, |         open_rate=2.0, | ||||||
|         close_profit=-0.2, |  | ||||||
|         close_profit_abs=-4.0, |  | ||||||
|         exchange='binance', |         exchange='binance', | ||||||
|         strategy='SampleStrategy', |         open_order_id=f'1234_{direc(is_short)}', | ||||||
|         open_order_id="prod_sell_6", |         strategy='StrategyTestV2', | ||||||
|         timeframe=5, |         timeframe=5, | ||||||
|  |         is_short=is_short, | ||||||
|     ) |     ) | ||||||
|     o = Order.parse_from_ccxt_object(mock_order_usdt_7(), 'LTC/USDT', 'buy') |     o = Order.parse_from_ccxt_object(mock_order_usdt_7(is_short), 'ADA/USDT', entry_side(is_short)) | ||||||
|     trade.orders.append(o) |  | ||||||
|     o = Order.parse_from_ccxt_object(mock_order_usdt_7_sell(), 'LTC/USDT', 'sell') |  | ||||||
|     trade.orders.append(o) |     trade.orders.append(o) | ||||||
|     return trade |     return trade | ||||||
|   | |||||||
| @@ -85,7 +85,7 @@ def test_load_backtest_data_new_format(testdatadir): | |||||||
|     filename = testdatadir / "backtest_results/backtest-result_new.json" |     filename = testdatadir / "backtest_results/backtest-result_new.json" | ||||||
|     bt_data = load_backtest_data(filename) |     bt_data = load_backtest_data(filename) | ||||||
|     assert isinstance(bt_data, DataFrame) |     assert isinstance(bt_data, DataFrame) | ||||||
|     assert set(bt_data.columns) == set(BT_DATA_COLUMNS + ['close_timestamp', 'open_timestamp']) |     assert set(bt_data.columns) == set(BT_DATA_COLUMNS) | ||||||
|     assert len(bt_data) == 179 |     assert len(bt_data) == 179 | ||||||
|  |  | ||||||
|     # Test loading from string (must yield same result) |     # Test loading from string (must yield same result) | ||||||
| @@ -110,7 +110,7 @@ def test_load_backtest_data_multi(testdatadir): | |||||||
|         bt_data = load_backtest_data(filename, strategy=strategy) |         bt_data = load_backtest_data(filename, strategy=strategy) | ||||||
|         assert isinstance(bt_data, DataFrame) |         assert isinstance(bt_data, DataFrame) | ||||||
|         assert set(bt_data.columns) == set( |         assert set(bt_data.columns) == set( | ||||||
|             BT_DATA_COLUMNS + ['close_timestamp', 'open_timestamp']) |             BT_DATA_COLUMNS) | ||||||
|         assert len(bt_data) == 179 |         assert len(bt_data) == 179 | ||||||
|  |  | ||||||
|         # Test loading from string (must yield same result) |         # Test loading from string (must yield same result) | ||||||
|   | |||||||
							
								
								
									
										191
									
								
								tests/data/test_entryexitanalysis.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										191
									
								
								tests/data/test_entryexitanalysis.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,191 @@ | |||||||
|  | import logging | ||||||
|  | from unittest.mock import MagicMock, PropertyMock | ||||||
|  |  | ||||||
|  | import pandas as pd | ||||||
|  | import pytest | ||||||
|  |  | ||||||
|  | from freqtrade.commands.analyze_commands import start_analysis_entries_exits | ||||||
|  | from freqtrade.commands.optimize_commands import start_backtesting | ||||||
|  | from freqtrade.enums import ExitType | ||||||
|  | from freqtrade.optimize.backtesting import Backtesting | ||||||
|  | from tests.conftest import get_args, patch_exchange, patched_configuration_load_config_file | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.fixture(autouse=True) | ||||||
|  | def entryexitanalysis_cleanup() -> None: | ||||||
|  |     yield None | ||||||
|  |  | ||||||
|  |     Backtesting.cleanup() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_backtest_analysis_nomock(default_conf, mocker, caplog, testdatadir, tmpdir, capsys): | ||||||
|  |     caplog.set_level(logging.INFO) | ||||||
|  |  | ||||||
|  |     default_conf.update({ | ||||||
|  |         "use_exit_signal": True, | ||||||
|  |         "exit_profit_only": False, | ||||||
|  |         "exit_profit_offset": 0.0, | ||||||
|  |         "ignore_roi_if_entry_signal": False, | ||||||
|  |     }) | ||||||
|  |     patch_exchange(mocker) | ||||||
|  |     result1 = pd.DataFrame({'pair': ['ETH/BTC', 'LTC/BTC', 'ETH/BTC', 'LTC/BTC'], | ||||||
|  |                             'profit_ratio': [0.025, 0.05, -0.1, -0.05], | ||||||
|  |                             'profit_abs': [0.5, 2.0, -4.0, -2.0], | ||||||
|  |                             'open_date': pd.to_datetime(['2018-01-29 18:40:00', | ||||||
|  |                                                          '2018-01-30 03:30:00', | ||||||
|  |                                                          '2018-01-30 08:10:00', | ||||||
|  |                                                          '2018-01-31 13:30:00', ], utc=True | ||||||
|  |                                                         ), | ||||||
|  |                             'close_date': pd.to_datetime(['2018-01-29 20:45:00', | ||||||
|  |                                                           '2018-01-30 05:35:00', | ||||||
|  |                                                           '2018-01-30 09:10:00', | ||||||
|  |                                                           '2018-01-31 15:00:00', ], utc=True), | ||||||
|  |                             'trade_duration': [235, 40, 60, 90], | ||||||
|  |                             'is_open': [False, False, False, False], | ||||||
|  |                             'stake_amount': [0.01, 0.01, 0.01, 0.01], | ||||||
|  |                             'open_rate': [0.104445, 0.10302485, 0.10302485, 0.10302485], | ||||||
|  |                             'close_rate': [0.104969, 0.103541, 0.102041, 0.102541], | ||||||
|  |                             "is_short": [False, False, False, False], | ||||||
|  |                             'enter_tag': ["enter_tag_long_a", | ||||||
|  |                                           "enter_tag_long_b", | ||||||
|  |                                           "enter_tag_long_a", | ||||||
|  |                                           "enter_tag_long_b"], | ||||||
|  |                             'exit_reason': [ExitType.ROI, | ||||||
|  |                                             ExitType.EXIT_SIGNAL, | ||||||
|  |                                             ExitType.STOP_LOSS, | ||||||
|  |                                             ExitType.TRAILING_STOP_LOSS] | ||||||
|  |                             }) | ||||||
|  |  | ||||||
|  |     backtestmock = MagicMock(side_effect=[ | ||||||
|  |         { | ||||||
|  |             'results': result1, | ||||||
|  |             'config': default_conf, | ||||||
|  |             'locks': [], | ||||||
|  |             'rejected_signals': 20, | ||||||
|  |             'timedout_entry_orders': 0, | ||||||
|  |             'timedout_exit_orders': 0, | ||||||
|  |             'canceled_trade_entries': 0, | ||||||
|  |             'canceled_entry_orders': 0, | ||||||
|  |             'replaced_entry_orders': 0, | ||||||
|  |             'final_balance': 1000, | ||||||
|  |         } | ||||||
|  |     ]) | ||||||
|  |     mocker.patch('freqtrade.plugins.pairlistmanager.PairListManager.whitelist', | ||||||
|  |                  PropertyMock(return_value=['ETH/BTC', 'LTC/BTC', 'DASH/BTC'])) | ||||||
|  |     mocker.patch('freqtrade.optimize.backtesting.Backtesting.backtest', backtestmock) | ||||||
|  |  | ||||||
|  |     patched_configuration_load_config_file(mocker, default_conf) | ||||||
|  |  | ||||||
|  |     args = [ | ||||||
|  |         'backtesting', | ||||||
|  |         '--config', 'config.json', | ||||||
|  |         '--datadir', str(testdatadir), | ||||||
|  |         '--user-data-dir', str(tmpdir), | ||||||
|  |         '--timeframe', '5m', | ||||||
|  |         '--timerange', '1515560100-1517287800', | ||||||
|  |         '--export', 'signals', | ||||||
|  |         '--cache', 'none', | ||||||
|  |     ] | ||||||
|  |     args = get_args(args) | ||||||
|  |     start_backtesting(args) | ||||||
|  |  | ||||||
|  |     captured = capsys.readouterr() | ||||||
|  |     assert 'BACKTESTING REPORT' in captured.out | ||||||
|  |     assert 'EXIT REASON STATS' in captured.out | ||||||
|  |     assert 'LEFT OPEN TRADES REPORT' in captured.out | ||||||
|  |  | ||||||
|  |     base_args = [ | ||||||
|  |         'backtesting-analysis', | ||||||
|  |         '--config', 'config.json', | ||||||
|  |         '--datadir', str(testdatadir), | ||||||
|  |         '--user-data-dir', str(tmpdir), | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |     # test group 0 and indicator list | ||||||
|  |     args = get_args(base_args + | ||||||
|  |                     ['--analysis-groups', "0", | ||||||
|  |                      '--indicator-list', "close", "rsi", "profit_abs"] | ||||||
|  |                     ) | ||||||
|  |     start_analysis_entries_exits(args) | ||||||
|  |     captured = capsys.readouterr() | ||||||
|  |     assert 'LTC/BTC' in captured.out | ||||||
|  |     assert 'ETH/BTC' in captured.out | ||||||
|  |     assert 'enter_tag_long_a' in captured.out | ||||||
|  |     assert 'enter_tag_long_b' in captured.out | ||||||
|  |     assert 'exit_signal' in captured.out | ||||||
|  |     assert 'roi' in captured.out | ||||||
|  |     assert 'stop_loss' in captured.out | ||||||
|  |     assert 'trailing_stop_loss' in captured.out | ||||||
|  |     assert '0.5' in captured.out | ||||||
|  |     assert '-4' in captured.out | ||||||
|  |     assert '-2' in captured.out | ||||||
|  |     assert '-3.5' in captured.out | ||||||
|  |     assert '50' in captured.out | ||||||
|  |     assert '0' in captured.out | ||||||
|  |     assert '0.01616' in captured.out | ||||||
|  |     assert '34.049' in captured.out | ||||||
|  |     assert '0.104104' in captured.out | ||||||
|  |     assert '47.0996' in captured.out | ||||||
|  |  | ||||||
|  |     # test group 1 | ||||||
|  |     args = get_args(base_args + ['--analysis-groups', "1"]) | ||||||
|  |     start_analysis_entries_exits(args) | ||||||
|  |     captured = capsys.readouterr() | ||||||
|  |     assert 'enter_tag_long_a' in captured.out | ||||||
|  |     assert 'enter_tag_long_b' in captured.out | ||||||
|  |     assert 'total_profit_pct' in captured.out | ||||||
|  |     assert '-3.5' in captured.out | ||||||
|  |     assert '-1.75' in captured.out | ||||||
|  |     assert '-7.5' in captured.out | ||||||
|  |     assert '-3.75' in captured.out | ||||||
|  |     assert '0' in captured.out | ||||||
|  |  | ||||||
|  |     # test group 2 | ||||||
|  |     args = get_args(base_args + ['--analysis-groups', "2"]) | ||||||
|  |     start_analysis_entries_exits(args) | ||||||
|  |     captured = capsys.readouterr() | ||||||
|  |     assert 'enter_tag_long_a' in captured.out | ||||||
|  |     assert 'enter_tag_long_b' in captured.out | ||||||
|  |     assert 'exit_signal' in captured.out | ||||||
|  |     assert 'roi' in captured.out | ||||||
|  |     assert 'stop_loss' in captured.out | ||||||
|  |     assert 'trailing_stop_loss' in captured.out | ||||||
|  |     assert 'total_profit_pct' in captured.out | ||||||
|  |     assert '-10' in captured.out | ||||||
|  |     assert '-5' in captured.out | ||||||
|  |     assert '2.5' in captured.out | ||||||
|  |  | ||||||
|  |     # test group 3 | ||||||
|  |     args = get_args(base_args + ['--analysis-groups', "3"]) | ||||||
|  |     start_analysis_entries_exits(args) | ||||||
|  |     captured = capsys.readouterr() | ||||||
|  |     assert 'LTC/BTC' in captured.out | ||||||
|  |     assert 'ETH/BTC' in captured.out | ||||||
|  |     assert 'enter_tag_long_a' in captured.out | ||||||
|  |     assert 'enter_tag_long_b' in captured.out | ||||||
|  |     assert 'total_profit_pct' in captured.out | ||||||
|  |     assert '-7.5' in captured.out | ||||||
|  |     assert '-3.75' in captured.out | ||||||
|  |     assert '-1.75' in captured.out | ||||||
|  |     assert '0' in captured.out | ||||||
|  |     assert '2' in captured.out | ||||||
|  |  | ||||||
|  |     # test group 4 | ||||||
|  |     args = get_args(base_args + ['--analysis-groups', "4"]) | ||||||
|  |     start_analysis_entries_exits(args) | ||||||
|  |     captured = capsys.readouterr() | ||||||
|  |     assert 'LTC/BTC' in captured.out | ||||||
|  |     assert 'ETH/BTC' in captured.out | ||||||
|  |     assert 'enter_tag_long_a' in captured.out | ||||||
|  |     assert 'enter_tag_long_b' in captured.out | ||||||
|  |     assert 'exit_signal' in captured.out | ||||||
|  |     assert 'roi' in captured.out | ||||||
|  |     assert 'stop_loss' in captured.out | ||||||
|  |     assert 'trailing_stop_loss' in captured.out | ||||||
|  |     assert 'total_profit_pct' in captured.out | ||||||
|  |     assert '-10' in captured.out | ||||||
|  |     assert '-5' in captured.out | ||||||
|  |     assert '-4' in captured.out | ||||||
|  |     assert '0.5' in captured.out | ||||||
|  |     assert '1' in captured.out | ||||||
|  |     assert '2.5' in captured.out | ||||||
| @@ -795,10 +795,27 @@ def test_backtest_one(default_conf, fee, mocker, testdatadir) -> None: | |||||||
|          'is_open': [False, False], |          'is_open': [False, False], | ||||||
|          'enter_tag': [None, None], |          'enter_tag': [None, None], | ||||||
|          "is_short": [False, False], |          "is_short": [False, False], | ||||||
|  |          'open_timestamp': [1517251200000, 1517283000000], | ||||||
|  |          'close_timestamp': [1517265300000, 1517285400000], | ||||||
|  |          'orders': [ | ||||||
|  |             [ | ||||||
|  |                 {'amount': 0.00957442, 'safe_price': 0.104445, 'ft_order_side': 'buy', | ||||||
|  |                  'order_filled_timestamp': 1517251200000, 'ft_is_entry': True}, | ||||||
|  |                 {'amount': 0.00957442, 'safe_price': 0.10496853383458644, 'ft_order_side': 'sell', | ||||||
|  |                  'order_filled_timestamp': 1517265300000, 'ft_is_entry': False} | ||||||
|  |             ], [ | ||||||
|  |                 {'amount': 0.0097064, 'safe_price': 0.10302485, 'ft_order_side': 'buy', | ||||||
|  |                  'order_filled_timestamp': 1517283000000, 'ft_is_entry': True}, | ||||||
|  |                 {'amount': 0.0097064, 'safe_price': 0.10354126528822055, 'ft_order_side': 'sell', | ||||||
|  |                  'order_filled_timestamp': 1517285400000, 'ft_is_entry': False} | ||||||
|  |             ] | ||||||
|  |          ] | ||||||
|          }) |          }) | ||||||
|     pd.testing.assert_frame_equal(results, expected) |     pd.testing.assert_frame_equal(results, expected) | ||||||
|  |     assert 'orders' in results.columns | ||||||
|     data_pair = processed[pair] |     data_pair = processed[pair] | ||||||
|     for _, t in results.iterrows(): |     for _, t in results.iterrows(): | ||||||
|  |         assert len(t['orders']) == 2 | ||||||
|         ln = data_pair.loc[data_pair["date"] == t["open_date"]] |         ln = data_pair.loc[data_pair["date"] == t["open_date"]] | ||||||
|         # Check open trade rate alignes to open rate |         # Check open trade rate alignes to open rate | ||||||
|         assert ln is not None |         assert ln is not None | ||||||
|   | |||||||
| @@ -70,9 +70,14 @@ def test_backtest_position_adjustment(default_conf, fee, mocker, testdatadir) -> | |||||||
|          'is_open': [False, False], |          'is_open': [False, False], | ||||||
|          'enter_tag': [None, None], |          'enter_tag': [None, None], | ||||||
|          'is_short': [False, False], |          'is_short': [False, False], | ||||||
|  |          'open_timestamp': [1517251200000, 1517283000000], | ||||||
|  |          'close_timestamp': [1517265300000, 1517285400000], | ||||||
|          }) |          }) | ||||||
|     pd.testing.assert_frame_equal(results, expected) |     pd.testing.assert_frame_equal(results.drop(columns=['orders']), expected) | ||||||
|     data_pair = processed[pair] |     data_pair = processed[pair] | ||||||
|  |     assert len(results.iloc[0]['orders']) == 6 | ||||||
|  |     assert len(results.iloc[1]['orders']) == 2 | ||||||
|  |  | ||||||
|     for _, t in results.iterrows(): |     for _, t in results.iterrows(): | ||||||
|         ln = data_pair.loc[data_pair["date"] == t["open_date"]] |         ln = data_pair.loc[data_pair["date"] == t["open_date"]] | ||||||
|         # Check open trade rate alignes to open rate |         # Check open trade rate alignes to open rate | ||||||
|   | |||||||
| @@ -171,7 +171,7 @@ def test_generate_backtest_stats(default_conf, testdatadir, tmpdir): | |||||||
|     _backup_file(filename_last, copy_file=True) |     _backup_file(filename_last, copy_file=True) | ||||||
|     assert not filename.is_file() |     assert not filename.is_file() | ||||||
|  |  | ||||||
|     store_backtest_stats(filename, stats) |     store_backtest_stats(filename, stats, '2022_01_01_15_05_13') | ||||||
|  |  | ||||||
|     # get real Filename (it's btresult-<date>.json) |     # get real Filename (it's btresult-<date>.json) | ||||||
|     last_fn = get_latest_backtest_filename(filename_last.parent) |     last_fn = get_latest_backtest_filename(filename_last.parent) | ||||||
| @@ -194,7 +194,7 @@ def test_store_backtest_stats(testdatadir, mocker): | |||||||
|  |  | ||||||
|     dump_mock = mocker.patch('freqtrade.optimize.optimize_reports.file_dump_json') |     dump_mock = mocker.patch('freqtrade.optimize.optimize_reports.file_dump_json') | ||||||
|  |  | ||||||
|     store_backtest_stats(testdatadir, {'metadata': {}}) |     store_backtest_stats(testdatadir, {'metadata': {}}, '2022_01_01_15_05_13') | ||||||
|  |  | ||||||
|     assert dump_mock.call_count == 3 |     assert dump_mock.call_count == 3 | ||||||
|     assert isinstance(dump_mock.call_args_list[0][0][0], Path) |     assert isinstance(dump_mock.call_args_list[0][0][0], Path) | ||||||
| @@ -202,7 +202,7 @@ def test_store_backtest_stats(testdatadir, mocker): | |||||||
|  |  | ||||||
|     dump_mock.reset_mock() |     dump_mock.reset_mock() | ||||||
|     filename = testdatadir / 'testresult.json' |     filename = testdatadir / 'testresult.json' | ||||||
|     store_backtest_stats(filename, {'metadata': {}}) |     store_backtest_stats(filename, {'metadata': {}}, '2022_01_01_15_05_13') | ||||||
|     assert dump_mock.call_count == 3 |     assert dump_mock.call_count == 3 | ||||||
|     assert isinstance(dump_mock.call_args_list[0][0][0], Path) |     assert isinstance(dump_mock.call_args_list[0][0][0], Path) | ||||||
|     # result will be testdatadir / testresult-<timestamp>.json |     # result will be testdatadir / testresult-<timestamp>.json | ||||||
| @@ -216,7 +216,7 @@ def test_store_backtest_candles(testdatadir, mocker): | |||||||
|     candle_dict = {'DefStrat': {'UNITTEST/BTC': pd.DataFrame()}} |     candle_dict = {'DefStrat': {'UNITTEST/BTC': pd.DataFrame()}} | ||||||
|  |  | ||||||
|     # mock directory exporting |     # mock directory exporting | ||||||
|     store_backtest_signal_candles(testdatadir, candle_dict) |     store_backtest_signal_candles(testdatadir, candle_dict, '2022_01_01_15_05_13') | ||||||
|  |  | ||||||
|     assert dump_mock.call_count == 1 |     assert dump_mock.call_count == 1 | ||||||
|     assert isinstance(dump_mock.call_args_list[0][0][0], Path) |     assert isinstance(dump_mock.call_args_list[0][0][0], Path) | ||||||
| @@ -225,7 +225,7 @@ def test_store_backtest_candles(testdatadir, mocker): | |||||||
|     dump_mock.reset_mock() |     dump_mock.reset_mock() | ||||||
|     # mock file exporting |     # mock file exporting | ||||||
|     filename = Path(testdatadir / 'testresult') |     filename = Path(testdatadir / 'testresult') | ||||||
|     store_backtest_signal_candles(filename, candle_dict) |     store_backtest_signal_candles(filename, candle_dict, '2022_01_01_15_05_13') | ||||||
|     assert dump_mock.call_count == 1 |     assert dump_mock.call_count == 1 | ||||||
|     assert isinstance(dump_mock.call_args_list[0][0][0], Path) |     assert isinstance(dump_mock.call_args_list[0][0][0], Path) | ||||||
|     # result will be testdatadir / testresult-<timestamp>_signals.pkl |     # result will be testdatadir / testresult-<timestamp>_signals.pkl | ||||||
| @@ -238,7 +238,7 @@ def test_write_read_backtest_candles(tmpdir): | |||||||
|     candle_dict = {'DefStrat': {'UNITTEST/BTC': pd.DataFrame()}} |     candle_dict = {'DefStrat': {'UNITTEST/BTC': pd.DataFrame()}} | ||||||
|  |  | ||||||
|     # test directory exporting |     # test directory exporting | ||||||
|     stored_file = store_backtest_signal_candles(Path(tmpdir), candle_dict) |     stored_file = store_backtest_signal_candles(Path(tmpdir), candle_dict, '2022_01_01_15_05_13') | ||||||
|     scp = open(stored_file, "rb") |     scp = open(stored_file, "rb") | ||||||
|     pickled_signal_candles = joblib.load(scp) |     pickled_signal_candles = joblib.load(scp) | ||||||
|     scp.close() |     scp.close() | ||||||
| @@ -252,7 +252,7 @@ def test_write_read_backtest_candles(tmpdir): | |||||||
|  |  | ||||||
|     # test file exporting |     # test file exporting | ||||||
|     filename = Path(tmpdir / 'testresult') |     filename = Path(tmpdir / 'testresult') | ||||||
|     stored_file = store_backtest_signal_candles(filename, candle_dict) |     stored_file = store_backtest_signal_candles(filename, candle_dict, '2022_01_01_15_05_13') | ||||||
|     scp = open(stored_file, "rb") |     scp = open(stored_file, "rb") | ||||||
|     pickled_signal_candles = joblib.load(scp) |     pickled_signal_candles = joblib.load(scp) | ||||||
|     scp.close() |     scp.close() | ||||||
|   | |||||||
| @@ -762,8 +762,8 @@ def test_PerformanceFilter_keep_mid_order(mocker, default_conf_usdt, fee, caplog | |||||||
|     with time_machine.travel("2021-09-01 05:00:00 +00:00") as t: |     with time_machine.travel("2021-09-01 05:00:00 +00:00") as t: | ||||||
|         create_mock_trades_usdt(fee) |         create_mock_trades_usdt(fee) | ||||||
|         pm.refresh_pairlist() |         pm.refresh_pairlist() | ||||||
|         assert pm.whitelist == ['XRP/USDT', 'ETC/USDT', 'ETH/USDT', |         assert pm.whitelist == ['XRP/USDT', 'ETC/USDT', 'ETH/USDT', 'LTC/USDT', | ||||||
|                                 'NEO/USDT', 'TKN/USDT', 'ADA/USDT', 'LTC/USDT'] |                                 'NEO/USDT', 'TKN/USDT', 'ADA/USDT', ] | ||||||
|         # assert log_has_re(r'Removing pair .* since .* is below .*', caplog) |         # assert log_has_re(r'Removing pair .* since .* is below .*', caplog) | ||||||
|  |  | ||||||
|         # Move to "outside" of lookback window, so original sorting is restored. |         # Move to "outside" of lookback window, so original sorting is restored. | ||||||
|   | |||||||
| @@ -11,11 +11,11 @@ from freqtrade.edge import PairInfo | |||||||
| from freqtrade.enums import SignalDirection, State, TradingMode | from freqtrade.enums import SignalDirection, State, TradingMode | ||||||
| from freqtrade.exceptions import ExchangeError, InvalidOrderException, TemporaryError | from freqtrade.exceptions import ExchangeError, InvalidOrderException, TemporaryError | ||||||
| from freqtrade.persistence import Trade | from freqtrade.persistence import Trade | ||||||
| from freqtrade.persistence.models import Order |  | ||||||
| from freqtrade.persistence.pairlock_middleware import PairLocks | from freqtrade.persistence.pairlock_middleware import PairLocks | ||||||
| from freqtrade.rpc import RPC, RPCException | from freqtrade.rpc import RPC, RPCException | ||||||
| from freqtrade.rpc.fiat_convert import CryptoToFiatConverter | from freqtrade.rpc.fiat_convert import CryptoToFiatConverter | ||||||
| from tests.conftest import create_mock_trades, get_patched_freqtradebot, patch_get_signal | from tests.conftest import (create_mock_trades, create_mock_trades_usdt, get_patched_freqtradebot, | ||||||
|  |                             patch_get_signal) | ||||||
|  |  | ||||||
|  |  | ||||||
| # Functions for recurrent object patching | # Functions for recurrent object patching | ||||||
| @@ -284,8 +284,8 @@ def test_rpc_status_table(default_conf, ticker, fee, mocker) -> None: | |||||||
|     assert isnan(fiat_profit_sum) |     assert isnan(fiat_profit_sum) | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_rpc_daily_profit(default_conf, update, ticker, fee, | def test__rpc_timeunit_profit(default_conf_usdt, ticker, fee, | ||||||
|                           limit_buy_order, limit_sell_order, markets, mocker) -> None: |                               limit_buy_order, limit_sell_order, markets, mocker) -> None: | ||||||
|     mocker.patch('freqtrade.rpc.telegram.Telegram', MagicMock()) |     mocker.patch('freqtrade.rpc.telegram.Telegram', MagicMock()) | ||||||
|     mocker.patch.multiple( |     mocker.patch.multiple( | ||||||
|         'freqtrade.exchange.Exchange', |         'freqtrade.exchange.Exchange', | ||||||
| @@ -294,45 +294,35 @@ def test_rpc_daily_profit(default_conf, update, ticker, fee, | |||||||
|         markets=PropertyMock(return_value=markets) |         markets=PropertyMock(return_value=markets) | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     freqtradebot = get_patched_freqtradebot(mocker, default_conf) |     freqtradebot = get_patched_freqtradebot(mocker, default_conf_usdt) | ||||||
|     patch_get_signal(freqtradebot) |     create_mock_trades_usdt(fee) | ||||||
|     stake_currency = default_conf['stake_currency'] |  | ||||||
|     fiat_display_currency = default_conf['fiat_display_currency'] |     stake_currency = default_conf_usdt['stake_currency'] | ||||||
|  |     fiat_display_currency = default_conf_usdt['fiat_display_currency'] | ||||||
|  |  | ||||||
|     rpc = RPC(freqtradebot) |     rpc = RPC(freqtradebot) | ||||||
|     rpc._fiat_converter = CryptoToFiatConverter() |     rpc._fiat_converter = CryptoToFiatConverter() | ||||||
|     # Create some test data |  | ||||||
|     freqtradebot.enter_positions() |  | ||||||
|     trade = Trade.query.first() |  | ||||||
|     assert trade |  | ||||||
|  |  | ||||||
|     # Simulate buy & sell |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_buy_order, limit_buy_order['symbol'], 'buy') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_sell_order, limit_sell_order['symbol'], 'sell') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|     trade.close_date = datetime.utcnow() |  | ||||||
|     trade.is_open = False |  | ||||||
|  |  | ||||||
|     # Try valid data |     # Try valid data | ||||||
|     update.message.text = '/daily 2' |     days = rpc._rpc_timeunit_profit(7, stake_currency, fiat_display_currency) | ||||||
|     days = rpc._rpc_daily_profit(7, stake_currency, fiat_display_currency) |  | ||||||
|     assert len(days['data']) == 7 |     assert len(days['data']) == 7 | ||||||
|     assert days['stake_currency'] == default_conf['stake_currency'] |     assert days['stake_currency'] == default_conf_usdt['stake_currency'] | ||||||
|     assert days['fiat_display_currency'] == default_conf['fiat_display_currency'] |     assert days['fiat_display_currency'] == default_conf_usdt['fiat_display_currency'] | ||||||
|     for day in days['data']: |     for day in days['data']: | ||||||
|         # [datetime.date(2018, 1, 11), '0.00000000 BTC', '0.000 USD'] |         # {'date': datetime.date(2022, 6, 11), 'abs_profit': 13.8299999, | ||||||
|         assert (day['abs_profit'] == 0.0 or |         #  'starting_balance': 1055.37, 'rel_profit': 0.0131044, | ||||||
|                 day['abs_profit'] == 0.00006217) |         #  'fiat_value': 0.0, 'trade_count': 2} | ||||||
|  |         assert day['abs_profit'] in (0.0, pytest.approx(13.8299999), pytest.approx(-4.0)) | ||||||
|         assert (day['fiat_value'] == 0.0 or |         assert day['rel_profit'] in (0.0, pytest.approx(0.01310441), pytest.approx(-0.00377583)) | ||||||
|                 day['fiat_value'] == 0.76748865) |         assert day['trade_count'] in (0, 1, 2) | ||||||
|  |         assert day['starting_balance'] in (pytest.approx(1059.37), pytest.approx(1055.37)) | ||||||
|  |         assert day['fiat_value'] in (0.0, ) | ||||||
|     # ensure first day is current date |     # ensure first day is current date | ||||||
|     assert str(days['data'][0]['date']) == str(datetime.utcnow().date()) |     assert str(days['data'][0]['date']) == str(datetime.utcnow().date()) | ||||||
|  |  | ||||||
|     # Try invalid data |     # Try invalid data | ||||||
|     with pytest.raises(RPCException, match=r'.*must be an integer greater than 0*'): |     with pytest.raises(RPCException, match=r'.*must be an integer greater than 0*'): | ||||||
|         rpc._rpc_daily_profit(0, stake_currency, fiat_display_currency) |         rpc._rpc_timeunit_profit(0, stake_currency, fiat_display_currency) | ||||||
|  |  | ||||||
|  |  | ||||||
| @pytest.mark.parametrize('is_short', [True, False]) | @pytest.mark.parametrize('is_short', [True, False]) | ||||||
| @@ -416,13 +406,8 @@ def test_rpc_delete_trade(mocker, default_conf, fee, markets, caplog, is_short): | |||||||
|     assert stoploss_mock.call_count == 0 |     assert stoploss_mock.call_count == 0 | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_rpc_trade_statistics(default_conf, ticker, ticker_sell_up, fee, | def test_rpc_trade_statistics(default_conf_usdt, ticker, fee, mocker) -> None: | ||||||
|                               limit_buy_order, limit_sell_order, mocker) -> None: |     mocker.patch('freqtrade.rpc.rpc.CryptoToFiatConverter._find_price', return_value=1.1) | ||||||
|     mocker.patch.multiple( |  | ||||||
|         'freqtrade.rpc.fiat_convert.CoinGeckoAPI', |  | ||||||
|         get_price=MagicMock(return_value={'bitcoin': {'usd': 15000.0}}), |  | ||||||
|     ) |  | ||||||
|     mocker.patch('freqtrade.rpc.rpc.CryptoToFiatConverter._find_price', return_value=15000.0) |  | ||||||
|     mocker.patch('freqtrade.rpc.telegram.Telegram', MagicMock()) |     mocker.patch('freqtrade.rpc.telegram.Telegram', MagicMock()) | ||||||
|     mocker.patch.multiple( |     mocker.patch.multiple( | ||||||
|         'freqtrade.exchange.Exchange', |         'freqtrade.exchange.Exchange', | ||||||
| @@ -430,10 +415,9 @@ def test_rpc_trade_statistics(default_conf, ticker, ticker_sell_up, fee, | |||||||
|         get_fee=fee, |         get_fee=fee, | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     freqtradebot = get_patched_freqtradebot(mocker, default_conf) |     freqtradebot = get_patched_freqtradebot(mocker, default_conf_usdt) | ||||||
|     patch_get_signal(freqtradebot) |     stake_currency = default_conf_usdt['stake_currency'] | ||||||
|     stake_currency = default_conf['stake_currency'] |     fiat_display_currency = default_conf_usdt['fiat_display_currency'] | ||||||
|     fiat_display_currency = default_conf['fiat_display_currency'] |  | ||||||
|  |  | ||||||
|     rpc = RPC(freqtradebot) |     rpc = RPC(freqtradebot) | ||||||
|     rpc._fiat_converter = CryptoToFiatConverter() |     rpc._fiat_converter = CryptoToFiatConverter() | ||||||
| @@ -446,75 +430,40 @@ def test_rpc_trade_statistics(default_conf, ticker, ticker_sell_up, fee, | |||||||
|     assert res['latest_trade_timestamp'] == 0 |     assert res['latest_trade_timestamp'] == 0 | ||||||
|  |  | ||||||
|     # Create some test data |     # Create some test data | ||||||
|     freqtradebot.enter_positions() |     create_mock_trades_usdt(fee) | ||||||
|     trade = Trade.query.first() |  | ||||||
|     # Simulate fulfilled LIMIT_BUY order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_buy_order, limit_buy_order['symbol'], 'sell') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     # Update the ticker with a market going up |  | ||||||
|     mocker.patch.multiple( |  | ||||||
|         'freqtrade.exchange.Exchange', |  | ||||||
|         fetch_ticker=ticker_sell_up |  | ||||||
|     ) |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_sell_order, limit_sell_order['symbol'], 'sell') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|     trade.close_date = datetime.utcnow() |  | ||||||
|     trade.is_open = False |  | ||||||
|  |  | ||||||
|     freqtradebot.enter_positions() |  | ||||||
|     trade = Trade.query.first() |  | ||||||
|     # Simulate fulfilled LIMIT_BUY order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_buy_order, limit_buy_order['symbol'], 'buy') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     # Update the ticker with a market going up |  | ||||||
|     mocker.patch.multiple( |  | ||||||
|         'freqtrade.exchange.Exchange', |  | ||||||
|         fetch_ticker=ticker_sell_up |  | ||||||
|     ) |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_sell_order, limit_sell_order['symbol'], 'sell') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|     trade.close_date = datetime.utcnow() |  | ||||||
|     trade.is_open = False |  | ||||||
|  |  | ||||||
|     stats = rpc._rpc_trade_statistics(stake_currency, fiat_display_currency) |     stats = rpc._rpc_trade_statistics(stake_currency, fiat_display_currency) | ||||||
|     assert prec_satoshi(stats['profit_closed_coin'], 6.217e-05) |     assert pytest.approx(stats['profit_closed_coin']) == 9.83 | ||||||
|     assert prec_satoshi(stats['profit_closed_percent_mean'], 6.2) |     assert pytest.approx(stats['profit_closed_percent_mean']) == -1.67 | ||||||
|     assert prec_satoshi(stats['profit_closed_fiat'], 0.93255) |     assert pytest.approx(stats['profit_closed_fiat']) == 10.813 | ||||||
|     assert prec_satoshi(stats['profit_all_coin'], 5.802e-05) |     assert pytest.approx(stats['profit_all_coin']) == -77.45964918 | ||||||
|     assert prec_satoshi(stats['profit_all_percent_mean'], 2.89) |     assert pytest.approx(stats['profit_all_percent_mean']) == -57.86 | ||||||
|     assert prec_satoshi(stats['profit_all_fiat'], 0.8703) |     assert pytest.approx(stats['profit_all_fiat']) == -85.205614098 | ||||||
|     assert stats['trade_count'] == 2 |     assert stats['trade_count'] == 7 | ||||||
|     assert stats['first_trade_date'] == 'just now' |     assert stats['first_trade_date'] == '2 days ago' | ||||||
|     assert stats['latest_trade_date'] == 'just now' |     assert stats['latest_trade_date'] == '17 minutes ago' | ||||||
|     assert stats['avg_duration'] in ('0:00:00', '0:00:01', '0:00:02') |     assert stats['avg_duration'] in ('0:17:40') | ||||||
|     assert stats['best_pair'] == 'ETH/BTC' |     assert stats['best_pair'] == 'XRP/USDT' | ||||||
|     assert prec_satoshi(stats['best_rate'], 6.2) |     assert stats['best_rate'] == 10.0 | ||||||
|  |  | ||||||
|     # Test non-available pair |     # Test non-available pair | ||||||
|     mocker.patch('freqtrade.exchange.Exchange.get_rate', |     mocker.patch('freqtrade.exchange.Exchange.get_rate', | ||||||
|                  MagicMock(side_effect=ExchangeError("Pair 'ETH/BTC' not available"))) |                  MagicMock(side_effect=ExchangeError("Pair 'XRP/USDT' not available"))) | ||||||
|     stats = rpc._rpc_trade_statistics(stake_currency, fiat_display_currency) |     stats = rpc._rpc_trade_statistics(stake_currency, fiat_display_currency) | ||||||
|     assert stats['trade_count'] == 2 |     assert stats['trade_count'] == 7 | ||||||
|     assert stats['first_trade_date'] == 'just now' |     assert stats['first_trade_date'] == '2 days ago' | ||||||
|     assert stats['latest_trade_date'] == 'just now' |     assert stats['latest_trade_date'] == '17 minutes ago' | ||||||
|     assert stats['avg_duration'] in ('0:00:00', '0:00:01', '0:00:02') |     assert stats['avg_duration'] in ('0:17:40') | ||||||
|     assert stats['best_pair'] == 'ETH/BTC' |     assert stats['best_pair'] == 'XRP/USDT' | ||||||
|     assert prec_satoshi(stats['best_rate'], 6.2) |     assert stats['best_rate'] == 10.0 | ||||||
|     assert isnan(stats['profit_all_coin']) |     assert isnan(stats['profit_all_coin']) | ||||||
|  |  | ||||||
|  |  | ||||||
| # Test that rpc_trade_statistics can handle trades that lacks | # Test that rpc_trade_statistics can handle trades that lacks | ||||||
| # trade.open_rate (it is set to None) | # trade.open_rate (it is set to None) | ||||||
| def test_rpc_trade_statistics_closed(mocker, default_conf, ticker, fee, | def test_rpc_trade_statistics_closed(mocker, default_conf_usdt, ticker, fee): | ||||||
|                                      ticker_sell_up, limit_buy_order, limit_sell_order): |  | ||||||
|     mocker.patch.multiple( |  | ||||||
|         'freqtrade.rpc.fiat_convert.CoinGeckoAPI', |  | ||||||
|         get_price=MagicMock(return_value={'bitcoin': {'usd': 15000.0}}), |  | ||||||
|     ) |  | ||||||
|     mocker.patch('freqtrade.rpc.fiat_convert.CryptoToFiatConverter._find_price', |     mocker.patch('freqtrade.rpc.fiat_convert.CryptoToFiatConverter._find_price', | ||||||
|                  return_value=15000.0) |                  return_value=1.1) | ||||||
|     mocker.patch('freqtrade.rpc.telegram.Telegram', MagicMock()) |     mocker.patch('freqtrade.rpc.telegram.Telegram', MagicMock()) | ||||||
|     mocker.patch.multiple( |     mocker.patch.multiple( | ||||||
|         'freqtrade.exchange.Exchange', |         'freqtrade.exchange.Exchange', | ||||||
| @@ -522,46 +471,32 @@ def test_rpc_trade_statistics_closed(mocker, default_conf, ticker, fee, | |||||||
|         get_fee=fee, |         get_fee=fee, | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     freqtradebot = get_patched_freqtradebot(mocker, default_conf) |     freqtradebot = get_patched_freqtradebot(mocker, default_conf_usdt) | ||||||
|     patch_get_signal(freqtradebot) |     patch_get_signal(freqtradebot) | ||||||
|     stake_currency = default_conf['stake_currency'] |     stake_currency = default_conf_usdt['stake_currency'] | ||||||
|     fiat_display_currency = default_conf['fiat_display_currency'] |     fiat_display_currency = default_conf_usdt['fiat_display_currency'] | ||||||
|  |  | ||||||
|     rpc = RPC(freqtradebot) |     rpc = RPC(freqtradebot) | ||||||
|  |  | ||||||
|     # Create some test data |     # Create some test data | ||||||
|     freqtradebot.enter_positions() |     create_mock_trades_usdt(fee) | ||||||
|     trade = Trade.query.first() |  | ||||||
|     # Simulate fulfilled LIMIT_BUY order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_buy_order, limit_buy_order['symbol'], 'buy') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|     # Update the ticker with a market going up |  | ||||||
|     mocker.patch.multiple( |  | ||||||
|         'freqtrade.exchange.Exchange', |  | ||||||
|         fetch_ticker=ticker_sell_up, |  | ||||||
|         get_fee=fee |  | ||||||
|     ) |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_sell_order, limit_sell_order['symbol'], 'sell') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|     trade.close_date = datetime.utcnow() |  | ||||||
|     trade.is_open = False |  | ||||||
|  |  | ||||||
|     for trade in Trade.query.order_by(Trade.id).all(): |     for trade in Trade.query.order_by(Trade.id).all(): | ||||||
|         trade.open_rate = None |         trade.open_rate = None | ||||||
|  |  | ||||||
|     stats = rpc._rpc_trade_statistics(stake_currency, fiat_display_currency) |     stats = rpc._rpc_trade_statistics(stake_currency, fiat_display_currency) | ||||||
|     assert prec_satoshi(stats['profit_closed_coin'], 0) |     assert stats['profit_closed_coin'] == 0 | ||||||
|     assert prec_satoshi(stats['profit_closed_percent_mean'], 0) |     assert stats['profit_closed_percent_mean'] == 0 | ||||||
|     assert prec_satoshi(stats['profit_closed_fiat'], 0) |     assert stats['profit_closed_fiat'] == 0 | ||||||
|     assert prec_satoshi(stats['profit_all_coin'], 0) |     assert stats['profit_all_coin'] == 0 | ||||||
|     assert prec_satoshi(stats['profit_all_percent_mean'], 0) |     assert stats['profit_all_percent_mean'] == 0 | ||||||
|     assert prec_satoshi(stats['profit_all_fiat'], 0) |     assert stats['profit_all_fiat'] == 0 | ||||||
|     assert stats['trade_count'] == 1 |     assert stats['trade_count'] == 7 | ||||||
|     assert stats['first_trade_date'] == 'just now' |     assert stats['first_trade_date'] == '2 days ago' | ||||||
|     assert stats['latest_trade_date'] == 'just now' |     assert stats['latest_trade_date'] == '17 minutes ago' | ||||||
|     assert stats['avg_duration'] == '0:00:00' |     assert stats['avg_duration'] == '0:00:00' | ||||||
|     assert stats['best_pair'] == 'ETH/BTC' |     assert stats['best_pair'] == 'XRP/USDT' | ||||||
|     assert prec_satoshi(stats['best_rate'], 6.2) |     assert stats['best_rate'] == 10.0 | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_rpc_balance_handle_error(default_conf, mocker): | def test_rpc_balance_handle_error(default_conf, mocker): | ||||||
| @@ -913,8 +848,7 @@ def test_rpc_force_exit(default_conf, ticker, fee, mocker) -> None: | |||||||
|     assert cancel_order_mock.call_count == 3 |     assert cancel_order_mock.call_count == 3 | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_performance_handle(default_conf, ticker, limit_buy_order, fee, | def test_performance_handle(default_conf_usdt, ticker, fee, mocker) -> None: | ||||||
|                             limit_sell_order, mocker) -> None: |  | ||||||
|     mocker.patch('freqtrade.rpc.telegram.Telegram', MagicMock()) |     mocker.patch('freqtrade.rpc.telegram.Telegram', MagicMock()) | ||||||
|     mocker.patch.multiple( |     mocker.patch.multiple( | ||||||
|         'freqtrade.exchange.Exchange', |         'freqtrade.exchange.Exchange', | ||||||
| @@ -923,34 +857,21 @@ def test_performance_handle(default_conf, ticker, limit_buy_order, fee, | |||||||
|         get_fee=fee, |         get_fee=fee, | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     freqtradebot = get_patched_freqtradebot(mocker, default_conf) |     freqtradebot = get_patched_freqtradebot(mocker, default_conf_usdt) | ||||||
|     patch_get_signal(freqtradebot) |     patch_get_signal(freqtradebot) | ||||||
|     rpc = RPC(freqtradebot) |     rpc = RPC(freqtradebot) | ||||||
|  |  | ||||||
|     # Create some test data |     # Create some test data | ||||||
|     freqtradebot.enter_positions() |     create_mock_trades_usdt(fee) | ||||||
|     trade = Trade.query.first() |  | ||||||
|     assert trade |  | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_BUY order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_buy_order, limit_buy_order['symbol'], 'buy') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_SELL order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_sell_order, limit_sell_order['symbol'], 'sell') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     trade.close_date = datetime.utcnow() |  | ||||||
|     trade.is_open = False |  | ||||||
|     res = rpc._rpc_performance() |     res = rpc._rpc_performance() | ||||||
|     assert len(res) == 1 |     assert len(res) == 3 | ||||||
|     assert res[0]['pair'] == 'ETH/BTC' |     assert res[0]['pair'] == 'XRP/USDT' | ||||||
|     assert res[0]['count'] == 1 |     assert res[0]['count'] == 1 | ||||||
|     assert prec_satoshi(res[0]['profit_pct'], 6.2) |     assert res[0]['profit_pct'] == 10.0 | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_enter_tag_performance_handle(default_conf, ticker, limit_buy_order, fee, | def test_enter_tag_performance_handle(default_conf, ticker, fee, mocker) -> None: | ||||||
|                                       limit_sell_order, mocker) -> None: |  | ||||||
|     mocker.patch('freqtrade.rpc.telegram.Telegram', MagicMock()) |     mocker.patch('freqtrade.rpc.telegram.Telegram', MagicMock()) | ||||||
|     mocker.patch.multiple( |     mocker.patch.multiple( | ||||||
|         'freqtrade.exchange.Exchange', |         'freqtrade.exchange.Exchange', | ||||||
| @@ -964,34 +885,22 @@ def test_enter_tag_performance_handle(default_conf, ticker, limit_buy_order, fee | |||||||
|     rpc = RPC(freqtradebot) |     rpc = RPC(freqtradebot) | ||||||
|  |  | ||||||
|     # Create some test data |     # Create some test data | ||||||
|  |     create_mock_trades_usdt(fee) | ||||||
|     freqtradebot.enter_positions() |     freqtradebot.enter_positions() | ||||||
|     trade = Trade.query.first() |  | ||||||
|     assert trade |  | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_BUY order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_buy_order, limit_buy_order['symbol'], 'buy') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_SELL order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_sell_order, limit_sell_order['symbol'], 'sell') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     trade.close_date = datetime.utcnow() |  | ||||||
|     trade.is_open = False |  | ||||||
|     res = rpc._rpc_enter_tag_performance(None) |     res = rpc._rpc_enter_tag_performance(None) | ||||||
|  |  | ||||||
|     assert len(res) == 1 |     assert len(res) == 3 | ||||||
|     assert res[0]['enter_tag'] == 'Other' |     assert res[0]['enter_tag'] == 'TEST3' | ||||||
|     assert res[0]['count'] == 1 |     assert res[0]['count'] == 1 | ||||||
|     assert prec_satoshi(res[0]['profit_pct'], 6.2) |     assert res[0]['profit_pct'] == 10.0 | ||||||
|  |  | ||||||
|     trade.enter_tag = "TEST_TAG" |  | ||||||
|     res = rpc._rpc_enter_tag_performance(None) |     res = rpc._rpc_enter_tag_performance(None) | ||||||
|  |  | ||||||
|     assert len(res) == 1 |     assert len(res) == 3 | ||||||
|     assert res[0]['enter_tag'] == 'TEST_TAG' |     assert res[0]['enter_tag'] == 'TEST3' | ||||||
|     assert res[0]['count'] == 1 |     assert res[0]['count'] == 1 | ||||||
|     assert prec_satoshi(res[0]['profit_pct'], 6.2) |     assert res[0]['profit_pct'] == 10.0 | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_enter_tag_performance_handle_2(mocker, default_conf, markets, fee): | def test_enter_tag_performance_handle_2(mocker, default_conf, markets, fee): | ||||||
| @@ -1023,8 +932,7 @@ def test_enter_tag_performance_handle_2(mocker, default_conf, markets, fee): | |||||||
|     assert prec_satoshi(res[0]['profit_pct'], 0.5) |     assert prec_satoshi(res[0]['profit_pct'], 0.5) | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_exit_reason_performance_handle(default_conf, ticker, limit_buy_order, fee, | def test_exit_reason_performance_handle(default_conf_usdt, ticker, fee, mocker) -> None: | ||||||
|                                         limit_sell_order, mocker) -> None: |  | ||||||
|     mocker.patch('freqtrade.rpc.telegram.Telegram', MagicMock()) |     mocker.patch('freqtrade.rpc.telegram.Telegram', MagicMock()) | ||||||
|     mocker.patch.multiple( |     mocker.patch.multiple( | ||||||
|         'freqtrade.exchange.Exchange', |         'freqtrade.exchange.Exchange', | ||||||
| @@ -1033,39 +941,22 @@ def test_exit_reason_performance_handle(default_conf, ticker, limit_buy_order, f | |||||||
|         get_fee=fee, |         get_fee=fee, | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     freqtradebot = get_patched_freqtradebot(mocker, default_conf) |     freqtradebot = get_patched_freqtradebot(mocker, default_conf_usdt) | ||||||
|     patch_get_signal(freqtradebot) |     patch_get_signal(freqtradebot) | ||||||
|     rpc = RPC(freqtradebot) |     rpc = RPC(freqtradebot) | ||||||
|  |  | ||||||
|     # Create some test data |     # Create some test data | ||||||
|     freqtradebot.enter_positions() |     create_mock_trades_usdt(fee) | ||||||
|     trade = Trade.query.first() |  | ||||||
|     assert trade |  | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_BUY order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_buy_order, limit_buy_order['symbol'], 'buy') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_SELL order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_sell_order, limit_sell_order['symbol'], 'sell') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     trade.close_date = datetime.utcnow() |  | ||||||
|     trade.is_open = False |  | ||||||
|     res = rpc._rpc_exit_reason_performance(None) |     res = rpc._rpc_exit_reason_performance(None) | ||||||
|  |  | ||||||
|     assert len(res) == 1 |     assert len(res) == 3 | ||||||
|     assert res[0]['exit_reason'] == 'Other' |     assert res[0]['exit_reason'] == 'roi' | ||||||
|     assert res[0]['count'] == 1 |     assert res[0]['count'] == 1 | ||||||
|     assert prec_satoshi(res[0]['profit_pct'], 6.2) |     assert res[0]['profit_pct'] == 10.0 | ||||||
|  |  | ||||||
|     trade.exit_reason = "TEST1" |     assert res[1]['exit_reason'] == 'exit_signal' | ||||||
|     res = rpc._rpc_exit_reason_performance(None) |     assert res[2]['exit_reason'] == 'Other' | ||||||
|  |  | ||||||
|     assert len(res) == 1 |  | ||||||
|     assert res[0]['exit_reason'] == 'TEST1' |  | ||||||
|     assert res[0]['count'] == 1 |  | ||||||
|     assert prec_satoshi(res[0]['profit_pct'], 6.2) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_exit_reason_performance_handle_2(mocker, default_conf, markets, fee): | def test_exit_reason_performance_handle_2(mocker, default_conf, markets, fee): | ||||||
| @@ -1097,8 +988,7 @@ def test_exit_reason_performance_handle_2(mocker, default_conf, markets, fee): | |||||||
|     assert prec_satoshi(res[0]['profit_pct'], 0.5) |     assert prec_satoshi(res[0]['profit_pct'], 0.5) | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_mix_tag_performance_handle(default_conf, ticker, limit_buy_order, fee, | def test_mix_tag_performance_handle(default_conf, ticker, fee, mocker) -> None: | ||||||
|                                     limit_sell_order, mocker) -> None: |  | ||||||
|     mocker.patch('freqtrade.rpc.telegram.Telegram', MagicMock()) |     mocker.patch('freqtrade.rpc.telegram.Telegram', MagicMock()) | ||||||
|     mocker.patch.multiple( |     mocker.patch.multiple( | ||||||
|         'freqtrade.exchange.Exchange', |         'freqtrade.exchange.Exchange', | ||||||
| @@ -1112,35 +1002,14 @@ def test_mix_tag_performance_handle(default_conf, ticker, limit_buy_order, fee, | |||||||
|     rpc = RPC(freqtradebot) |     rpc = RPC(freqtradebot) | ||||||
|  |  | ||||||
|     # Create some test data |     # Create some test data | ||||||
|     freqtradebot.enter_positions() |     create_mock_trades_usdt(fee) | ||||||
|     trade = Trade.query.first() |  | ||||||
|     assert trade |  | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_BUY order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_buy_order, limit_buy_order['symbol'], 'buy') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_SELL order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_sell_order, limit_sell_order['symbol'], 'sell') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     trade.close_date = datetime.utcnow() |  | ||||||
|     trade.is_open = False |  | ||||||
|     res = rpc._rpc_mix_tag_performance(None) |     res = rpc._rpc_mix_tag_performance(None) | ||||||
|  |  | ||||||
|     assert len(res) == 1 |     assert len(res) == 3 | ||||||
|     assert res[0]['mix_tag'] == 'Other Other' |     assert res[0]['mix_tag'] == 'TEST3 roi' | ||||||
|     assert res[0]['count'] == 1 |     assert res[0]['count'] == 1 | ||||||
|     assert prec_satoshi(res[0]['profit_pct'], 6.2) |     assert res[0]['profit_pct'] == 10.0 | ||||||
|  |  | ||||||
|     trade.enter_tag = "TESTBUY" |  | ||||||
|     trade.exit_reason = "TESTSELL" |  | ||||||
|     res = rpc._rpc_mix_tag_performance(None) |  | ||||||
|  |  | ||||||
|     assert len(res) == 1 |  | ||||||
|     assert res[0]['mix_tag'] == 'TESTBUY TESTSELL' |  | ||||||
|     assert res[0]['count'] == 1 |  | ||||||
|     assert prec_satoshi(res[0]['profit_pct'], 6.2) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_mix_tag_performance_handle_2(mocker, default_conf, markets, fee): | def test_mix_tag_performance_handle_2(mocker, default_conf, markets, fee): | ||||||
|   | |||||||
| @@ -1384,12 +1384,14 @@ def test_api_strategies(botclient): | |||||||
|     rc = client_get(client, f"{BASE_URI}/strategies") |     rc = client_get(client, f"{BASE_URI}/strategies") | ||||||
|  |  | ||||||
|     assert_response(rc) |     assert_response(rc) | ||||||
|  |  | ||||||
|     assert rc.json() == {'strategies': [ |     assert rc.json() == {'strategies': [ | ||||||
|         'HyperoptableStrategy', |         'HyperoptableStrategy', | ||||||
|         'InformativeDecoratorTest', |         'InformativeDecoratorTest', | ||||||
|         'StrategyTestV2', |         'StrategyTestV2', | ||||||
|         'StrategyTestV3', |         'StrategyTestV3', | ||||||
|         'StrategyTestV3Futures', |         'StrategyTestV3Analysis', | ||||||
|  |         'StrategyTestV3Futures' | ||||||
|     ]} |     ]} | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -27,8 +27,9 @@ from freqtrade.persistence.models import Order | |||||||
| from freqtrade.rpc import RPC | from freqtrade.rpc import RPC | ||||||
| from freqtrade.rpc.rpc import RPCException | from freqtrade.rpc.rpc import RPCException | ||||||
| from freqtrade.rpc.telegram import Telegram, authorized_only | from freqtrade.rpc.telegram import Telegram, authorized_only | ||||||
| from tests.conftest import (CURRENT_TEST_STRATEGY, create_mock_trades, get_patched_freqtradebot, | from tests.conftest import (CURRENT_TEST_STRATEGY, create_mock_trades, create_mock_trades_usdt, | ||||||
|                             log_has, log_has_re, patch_exchange, patch_get_signal, patch_whitelist) |                             get_patched_freqtradebot, log_has, log_has_re, patch_exchange, | ||||||
|  |                             patch_get_signal, patch_whitelist) | ||||||
|  |  | ||||||
|  |  | ||||||
| class DummyCls(Telegram): | class DummyCls(Telegram): | ||||||
| @@ -404,12 +405,10 @@ def test_status_table_handle(default_conf, update, ticker, fee, mocker) -> None: | |||||||
|     assert msg_mock.call_count == 1 |     assert msg_mock.call_count == 1 | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_daily_handle(default_conf, update, ticker, limit_buy_order, fee, | def test_daily_handle(default_conf_usdt, update, ticker, fee, mocker, time_machine) -> None: | ||||||
|                       limit_sell_order, mocker) -> None: |  | ||||||
|     default_conf['max_open_trades'] = 1 |  | ||||||
|     mocker.patch( |     mocker.patch( | ||||||
|         'freqtrade.rpc.rpc.CryptoToFiatConverter._find_price', |         'freqtrade.rpc.rpc.CryptoToFiatConverter._find_price', | ||||||
|         return_value=15000.0 |         return_value=1.1 | ||||||
|     ) |     ) | ||||||
|     mocker.patch.multiple( |     mocker.patch.multiple( | ||||||
|         'freqtrade.exchange.Exchange', |         'freqtrade.exchange.Exchange', | ||||||
| @@ -417,25 +416,12 @@ def test_daily_handle(default_conf, update, ticker, limit_buy_order, fee, | |||||||
|         get_fee=fee, |         get_fee=fee, | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     telegram, freqtradebot, msg_mock = get_telegram_testobject(mocker, default_conf) |     telegram, freqtradebot, msg_mock = get_telegram_testobject(mocker, default_conf_usdt) | ||||||
|  |  | ||||||
|     patch_get_signal(freqtradebot) |  | ||||||
|  |  | ||||||
|  |     # Move date to within day | ||||||
|  |     time_machine.move_to('2022-06-11 08:00:00+00:00') | ||||||
|     # Create some test data |     # Create some test data | ||||||
|     freqtradebot.enter_positions() |     create_mock_trades_usdt(fee) | ||||||
|     trade = Trade.query.first() |  | ||||||
|     assert trade |  | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_BUY order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_buy_order, limit_buy_order['symbol'], 'buy') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_SELL order for trade |  | ||||||
|     oobjs = Order.parse_from_ccxt_object(limit_sell_order, limit_sell_order['symbol'], 'sell') |  | ||||||
|     trade.update_trade(oobjs) |  | ||||||
|  |  | ||||||
|     trade.close_date = datetime.utcnow() |  | ||||||
|     trade.is_open = False |  | ||||||
|  |  | ||||||
|     # Try valid data |     # Try valid data | ||||||
|     # /daily 2 |     # /daily 2 | ||||||
| @@ -446,10 +432,11 @@ def test_daily_handle(default_conf, update, ticker, limit_buy_order, fee, | |||||||
|     assert "Daily Profit over the last 2 days</b>:" in msg_mock.call_args_list[0][0][0] |     assert "Daily Profit over the last 2 days</b>:" in msg_mock.call_args_list[0][0][0] | ||||||
|     assert 'Day ' in msg_mock.call_args_list[0][0][0] |     assert 'Day ' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str(datetime.utcnow().date()) in msg_mock.call_args_list[0][0][0] |     assert str(datetime.utcnow().date()) in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  0.00006217 BTC') in msg_mock.call_args_list[0][0][0] |     assert '  13.83 USDT' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  0.933 USD') in msg_mock.call_args_list[0][0][0] |     assert '  15.21 USD' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  1 trade') in msg_mock.call_args_list[0][0][0] |     assert '(2)' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  0 trade') in msg_mock.call_args_list[0][0][0] |     assert '(2)  13.83 USDT  15.21 USD  1.31%' in msg_mock.call_args_list[0][0][0] | ||||||
|  |     assert '(0)' in msg_mock.call_args_list[0][0][0] | ||||||
|  |  | ||||||
|     # Reset msg_mock |     # Reset msg_mock | ||||||
|     msg_mock.reset_mock() |     msg_mock.reset_mock() | ||||||
| @@ -458,32 +445,23 @@ def test_daily_handle(default_conf, update, ticker, limit_buy_order, fee, | |||||||
|     assert msg_mock.call_count == 1 |     assert msg_mock.call_count == 1 | ||||||
|     assert "Daily Profit over the last 7 days</b>:" in msg_mock.call_args_list[0][0][0] |     assert "Daily Profit over the last 7 days</b>:" in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str(datetime.utcnow().date()) in msg_mock.call_args_list[0][0][0] |     assert str(datetime.utcnow().date()) in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  0.00006217 BTC') in msg_mock.call_args_list[0][0][0] |     assert str((datetime.utcnow() - timedelta(days=5)).date()) in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  0.933 USD') in msg_mock.call_args_list[0][0][0] |     assert '  13.83 USDT' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  1 trade') in msg_mock.call_args_list[0][0][0] |     assert '  15.21 USD' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  0 trade') in msg_mock.call_args_list[0][0][0] |     assert '(2)' in msg_mock.call_args_list[0][0][0] | ||||||
|  |     assert '(1)' in msg_mock.call_args_list[0][0][0] | ||||||
|  |     assert '(0)' in msg_mock.call_args_list[0][0][0] | ||||||
|  |  | ||||||
|     # Reset msg_mock |     # Reset msg_mock | ||||||
|     msg_mock.reset_mock() |     msg_mock.reset_mock() | ||||||
|     freqtradebot.config['max_open_trades'] = 2 |  | ||||||
|     # Add two other trades |  | ||||||
|     n = freqtradebot.enter_positions() |  | ||||||
|     assert n == 2 |  | ||||||
|  |  | ||||||
|     trades = Trade.query.all() |  | ||||||
|     for trade in trades: |  | ||||||
|         trade.update_trade(oobj) |  | ||||||
|         trade.update_trade(oobjs) |  | ||||||
|         trade.close_date = datetime.utcnow() |  | ||||||
|         trade.is_open = False |  | ||||||
|  |  | ||||||
|     # /daily 1 |     # /daily 1 | ||||||
|     context = MagicMock() |     context = MagicMock() | ||||||
|     context.args = ["1"] |     context.args = ["1"] | ||||||
|     telegram._daily(update=update, context=context) |     telegram._daily(update=update, context=context) | ||||||
|     assert str('  0.00018651 BTC') in msg_mock.call_args_list[0][0][0] |     assert '  13.83 USDT' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  2.798 USD') in msg_mock.call_args_list[0][0][0] |     assert '  15.21 USD' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  3 trades') in msg_mock.call_args_list[0][0][0] |     assert '(2)' in msg_mock.call_args_list[0][0][0] | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_daily_wrong_input(default_conf, update, ticker, mocker) -> None: | def test_daily_wrong_input(default_conf, update, ticker, mocker) -> None: | ||||||
| @@ -512,15 +490,14 @@ def test_daily_wrong_input(default_conf, update, ticker, mocker) -> None: | |||||||
|     context = MagicMock() |     context = MagicMock() | ||||||
|     context.args = ["today"] |     context.args = ["today"] | ||||||
|     telegram._daily(update=update, context=context) |     telegram._daily(update=update, context=context) | ||||||
|     assert str('Daily Profit over the last 7 days</b>:') in msg_mock.call_args_list[0][0][0] |     assert 'Daily Profit over the last 7 days</b>:' in msg_mock.call_args_list[0][0][0] | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_weekly_handle(default_conf, update, ticker, limit_buy_order, fee, | def test_weekly_handle(default_conf_usdt, update, ticker, fee, mocker, time_machine) -> None: | ||||||
|                        limit_sell_order, mocker) -> None: |     default_conf_usdt['max_open_trades'] = 1 | ||||||
|     default_conf['max_open_trades'] = 1 |  | ||||||
|     mocker.patch( |     mocker.patch( | ||||||
|         'freqtrade.rpc.rpc.CryptoToFiatConverter._find_price', |         'freqtrade.rpc.rpc.CryptoToFiatConverter._find_price', | ||||||
|         return_value=15000.0 |         return_value=1.1 | ||||||
|     ) |     ) | ||||||
|     mocker.patch.multiple( |     mocker.patch.multiple( | ||||||
|         'freqtrade.exchange.Exchange', |         'freqtrade.exchange.Exchange', | ||||||
| @@ -528,25 +505,10 @@ def test_weekly_handle(default_conf, update, ticker, limit_buy_order, fee, | |||||||
|         get_fee=fee, |         get_fee=fee, | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     telegram, freqtradebot, msg_mock = get_telegram_testobject(mocker, default_conf) |     telegram, freqtradebot, msg_mock = get_telegram_testobject(mocker, default_conf_usdt) | ||||||
|  |     # Move to saturday - so all trades are within that week | ||||||
|     patch_get_signal(freqtradebot) |     time_machine.move_to('2022-06-11') | ||||||
|  |     create_mock_trades_usdt(fee) | ||||||
|     # Create some test data |  | ||||||
|     freqtradebot.enter_positions() |  | ||||||
|     trade = Trade.query.first() |  | ||||||
|     assert trade |  | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_BUY order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_buy_order, limit_buy_order['symbol'], 'buy') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_SELL order for trade |  | ||||||
|     oobjs = Order.parse_from_ccxt_object(limit_sell_order, limit_sell_order['symbol'], 'sell') |  | ||||||
|     trade.update_trade(oobjs) |  | ||||||
|  |  | ||||||
|     trade.close_date = datetime.utcnow() |  | ||||||
|     trade.is_open = False |  | ||||||
|  |  | ||||||
|     # Try valid data |     # Try valid data | ||||||
|     # /weekly 2 |     # /weekly 2 | ||||||
| @@ -560,10 +522,10 @@ def test_weekly_handle(default_conf, update, ticker, limit_buy_order, fee, | |||||||
|     today = datetime.utcnow().date() |     today = datetime.utcnow().date() | ||||||
|     first_iso_day_of_current_week = today - timedelta(days=today.weekday()) |     first_iso_day_of_current_week = today - timedelta(days=today.weekday()) | ||||||
|     assert str(first_iso_day_of_current_week) in msg_mock.call_args_list[0][0][0] |     assert str(first_iso_day_of_current_week) in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  0.00006217 BTC') in msg_mock.call_args_list[0][0][0] |     assert '  9.83 USDT' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  0.933 USD') in msg_mock.call_args_list[0][0][0] |     assert '  10.81 USD' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  1 trade') in msg_mock.call_args_list[0][0][0] |     assert '(3)' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  0 trade') in msg_mock.call_args_list[0][0][0] |     assert '(0)' in msg_mock.call_args_list[0][0][0] | ||||||
|  |  | ||||||
|     # Reset msg_mock |     # Reset msg_mock | ||||||
|     msg_mock.reset_mock() |     msg_mock.reset_mock() | ||||||
| @@ -573,44 +535,10 @@ def test_weekly_handle(default_conf, update, ticker, limit_buy_order, fee, | |||||||
|     assert "Weekly Profit over the last 8 weeks (starting from Monday)</b>:" \ |     assert "Weekly Profit over the last 8 weeks (starting from Monday)</b>:" \ | ||||||
|            in msg_mock.call_args_list[0][0][0] |            in msg_mock.call_args_list[0][0][0] | ||||||
|     assert 'Weekly' in msg_mock.call_args_list[0][0][0] |     assert 'Weekly' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  0.00006217 BTC') in msg_mock.call_args_list[0][0][0] |     assert '  9.83 USDT' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  0.933 USD') in msg_mock.call_args_list[0][0][0] |     assert '  10.81 USD' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  1 trade') in msg_mock.call_args_list[0][0][0] |     assert '(3)' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  0 trade') in msg_mock.call_args_list[0][0][0] |     assert '(0)' in msg_mock.call_args_list[0][0][0] | ||||||
|  |  | ||||||
|     # Reset msg_mock |  | ||||||
|     msg_mock.reset_mock() |  | ||||||
|     freqtradebot.config['max_open_trades'] = 2 |  | ||||||
|     # Add two other trades |  | ||||||
|     n = freqtradebot.enter_positions() |  | ||||||
|     assert n == 2 |  | ||||||
|  |  | ||||||
|     trades = Trade.query.all() |  | ||||||
|     for trade in trades: |  | ||||||
|         trade.update_trade(oobj) |  | ||||||
|         trade.update_trade(oobjs) |  | ||||||
|         trade.close_date = datetime.utcnow() |  | ||||||
|         trade.is_open = False |  | ||||||
|  |  | ||||||
|     # /weekly 1 |  | ||||||
|     # By default, the 8 previous weeks are shown |  | ||||||
|     # So the previous modified trade should be excluded from the stats |  | ||||||
|     context = MagicMock() |  | ||||||
|     context.args = ["1"] |  | ||||||
|     telegram._weekly(update=update, context=context) |  | ||||||
|     assert str('  0.00018651 BTC') in msg_mock.call_args_list[0][0][0] |  | ||||||
|     assert str('  2.798 USD') in msg_mock.call_args_list[0][0][0] |  | ||||||
|     assert str('  3 trades') in msg_mock.call_args_list[0][0][0] |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_weekly_wrong_input(default_conf, update, ticker, mocker) -> None: |  | ||||||
|     mocker.patch.multiple( |  | ||||||
|         'freqtrade.exchange.Exchange', |  | ||||||
|         fetch_ticker=ticker |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     telegram, freqtradebot, msg_mock = get_telegram_testobject(mocker, default_conf) |  | ||||||
|     patch_get_signal(freqtradebot) |  | ||||||
|  |  | ||||||
|     # Try invalid data |     # Try invalid data | ||||||
|     msg_mock.reset_mock() |     msg_mock.reset_mock() | ||||||
| @@ -629,16 +557,17 @@ def test_weekly_wrong_input(default_conf, update, ticker, mocker) -> None: | |||||||
|     context = MagicMock() |     context = MagicMock() | ||||||
|     context.args = ["this week"] |     context.args = ["this week"] | ||||||
|     telegram._weekly(update=update, context=context) |     telegram._weekly(update=update, context=context) | ||||||
|     assert str('Weekly Profit over the last 8 weeks (starting from Monday)</b>:') \ |     assert ( | ||||||
|  |         'Weekly Profit over the last 8 weeks (starting from Monday)</b>:' | ||||||
|         in msg_mock.call_args_list[0][0][0] |         in msg_mock.call_args_list[0][0][0] | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_monthly_handle(default_conf, update, ticker, limit_buy_order, fee, | def test_monthly_handle(default_conf_usdt, update, ticker, fee, mocker, time_machine) -> None: | ||||||
|                         limit_sell_order, mocker) -> None: |     default_conf_usdt['max_open_trades'] = 1 | ||||||
|     default_conf['max_open_trades'] = 1 |  | ||||||
|     mocker.patch( |     mocker.patch( | ||||||
|         'freqtrade.rpc.rpc.CryptoToFiatConverter._find_price', |         'freqtrade.rpc.rpc.CryptoToFiatConverter._find_price', | ||||||
|         return_value=15000.0 |         return_value=1.1 | ||||||
|     ) |     ) | ||||||
|     mocker.patch.multiple( |     mocker.patch.multiple( | ||||||
|         'freqtrade.exchange.Exchange', |         'freqtrade.exchange.Exchange', | ||||||
| @@ -646,25 +575,10 @@ def test_monthly_handle(default_conf, update, ticker, limit_buy_order, fee, | |||||||
|         get_fee=fee, |         get_fee=fee, | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     telegram, freqtradebot, msg_mock = get_telegram_testobject(mocker, default_conf) |     telegram, freqtradebot, msg_mock = get_telegram_testobject(mocker, default_conf_usdt) | ||||||
|  |     # Move to day within the month so all mock trades fall into this week. | ||||||
|     patch_get_signal(freqtradebot) |     time_machine.move_to('2022-06-11') | ||||||
|  |     create_mock_trades_usdt(fee) | ||||||
|     # Create some test data |  | ||||||
|     freqtradebot.enter_positions() |  | ||||||
|     trade = Trade.query.first() |  | ||||||
|     assert trade |  | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_BUY order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_buy_order, limit_buy_order['symbol'], 'buy') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_SELL order for trade |  | ||||||
|     oobjs = Order.parse_from_ccxt_object(limit_sell_order, limit_sell_order['symbol'], 'sell') |  | ||||||
|     trade.update_trade(oobjs) |  | ||||||
|  |  | ||||||
|     trade.close_date = datetime.utcnow() |  | ||||||
|     trade.is_open = False |  | ||||||
|  |  | ||||||
|     # Try valid data |     # Try valid data | ||||||
|     # /monthly 2 |     # /monthly 2 | ||||||
| @@ -677,10 +591,10 @@ def test_monthly_handle(default_conf, update, ticker, limit_buy_order, fee, | |||||||
|     today = datetime.utcnow().date() |     today = datetime.utcnow().date() | ||||||
|     current_month = f"{today.year}-{today.month:02} " |     current_month = f"{today.year}-{today.month:02} " | ||||||
|     assert current_month in msg_mock.call_args_list[0][0][0] |     assert current_month in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  0.00006217 BTC') in msg_mock.call_args_list[0][0][0] |     assert '  9.83 USDT' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  0.933 USD') in msg_mock.call_args_list[0][0][0] |     assert '  10.81 USD' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  1 trade') in msg_mock.call_args_list[0][0][0] |     assert '(3)' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  0 trade') in msg_mock.call_args_list[0][0][0] |     assert '(0)' in msg_mock.call_args_list[0][0][0] | ||||||
|  |  | ||||||
|     # Reset msg_mock |     # Reset msg_mock | ||||||
|     msg_mock.reset_mock() |     msg_mock.reset_mock() | ||||||
| @@ -691,24 +605,13 @@ def test_monthly_handle(default_conf, update, ticker, limit_buy_order, fee, | |||||||
|     assert 'Monthly Profit over the last 6 months</b>:' in msg_mock.call_args_list[0][0][0] |     assert 'Monthly Profit over the last 6 months</b>:' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert 'Month ' in msg_mock.call_args_list[0][0][0] |     assert 'Month ' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert current_month in msg_mock.call_args_list[0][0][0] |     assert current_month in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  0.00006217 BTC') in msg_mock.call_args_list[0][0][0] |     assert '  9.83 USDT' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  0.933 USD') in msg_mock.call_args_list[0][0][0] |     assert '  10.81 USD' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  1 trade') in msg_mock.call_args_list[0][0][0] |     assert '(3)' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  0 trade') in msg_mock.call_args_list[0][0][0] |     assert '(0)' in msg_mock.call_args_list[0][0][0] | ||||||
|  |  | ||||||
|     # Reset msg_mock |     # Reset msg_mock | ||||||
|     msg_mock.reset_mock() |     msg_mock.reset_mock() | ||||||
|     freqtradebot.config['max_open_trades'] = 2 |  | ||||||
|     # Add two other trades |  | ||||||
|     n = freqtradebot.enter_positions() |  | ||||||
|     assert n == 2 |  | ||||||
|  |  | ||||||
|     trades = Trade.query.all() |  | ||||||
|     for trade in trades: |  | ||||||
|         trade.update_trade(oobj) |  | ||||||
|         trade.update_trade(oobjs) |  | ||||||
|         trade.close_date = datetime.utcnow() |  | ||||||
|         trade.is_open = False |  | ||||||
|  |  | ||||||
|     # /monthly 12 |     # /monthly 12 | ||||||
|     context = MagicMock() |     context = MagicMock() | ||||||
| @@ -716,24 +619,14 @@ def test_monthly_handle(default_conf, update, ticker, limit_buy_order, fee, | |||||||
|     telegram._monthly(update=update, context=context) |     telegram._monthly(update=update, context=context) | ||||||
|     assert msg_mock.call_count == 1 |     assert msg_mock.call_count == 1 | ||||||
|     assert 'Monthly Profit over the last 12 months</b>:' in msg_mock.call_args_list[0][0][0] |     assert 'Monthly Profit over the last 12 months</b>:' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  0.00018651 BTC') in msg_mock.call_args_list[0][0][0] |     assert '  9.83 USDT' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  2.798 USD') in msg_mock.call_args_list[0][0][0] |     assert '  10.81 USD' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert str('  3 trades') in msg_mock.call_args_list[0][0][0] |     assert '(3)' in msg_mock.call_args_list[0][0][0] | ||||||
|  |  | ||||||
|     # The one-digit months should contain a zero, Eg: September 2021 = "2021-09" |     # The one-digit months should contain a zero, Eg: September 2021 = "2021-09" | ||||||
|     # Since we loaded the last 12 months, any month should appear |     # Since we loaded the last 12 months, any month should appear | ||||||
|     assert str('-09') in msg_mock.call_args_list[0][0][0] |     assert str('-09') in msg_mock.call_args_list[0][0][0] | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_monthly_wrong_input(default_conf, update, ticker, mocker) -> None: |  | ||||||
|     mocker.patch.multiple( |  | ||||||
|         'freqtrade.exchange.Exchange', |  | ||||||
|         fetch_ticker=ticker |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     telegram, freqtradebot, msg_mock = get_telegram_testobject(mocker, default_conf) |  | ||||||
|     patch_get_signal(freqtradebot) |  | ||||||
|  |  | ||||||
|     # Try invalid data |     # Try invalid data | ||||||
|     msg_mock.reset_mock() |     msg_mock.reset_mock() | ||||||
|     freqtradebot.state = State.RUNNING |     freqtradebot.state = State.RUNNING | ||||||
| @@ -754,16 +647,16 @@ def test_monthly_wrong_input(default_conf, update, ticker, mocker) -> None: | |||||||
|     assert str('Monthly Profit over the last 6 months</b>:') in msg_mock.call_args_list[0][0][0] |     assert str('Monthly Profit over the last 6 months</b>:') in msg_mock.call_args_list[0][0][0] | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_profit_handle(default_conf, update, ticker, ticker_sell_up, fee, | def test_profit_handle(default_conf_usdt, update, ticker_usdt, ticker_sell_up, fee, | ||||||
|                        limit_buy_order, limit_sell_order, mocker) -> None: |                        limit_sell_order_usdt, mocker) -> None: | ||||||
|     mocker.patch('freqtrade.rpc.rpc.CryptoToFiatConverter._find_price', return_value=15000.0) |     mocker.patch('freqtrade.rpc.rpc.CryptoToFiatConverter._find_price', return_value=1.1) | ||||||
|     mocker.patch.multiple( |     mocker.patch.multiple( | ||||||
|         'freqtrade.exchange.Exchange', |         'freqtrade.exchange.Exchange', | ||||||
|         fetch_ticker=ticker, |         fetch_ticker=ticker_usdt, | ||||||
|         get_fee=fee, |         get_fee=fee, | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     telegram, freqtradebot, msg_mock = get_telegram_testobject(mocker, default_conf) |     telegram, freqtradebot, msg_mock = get_telegram_testobject(mocker, default_conf_usdt) | ||||||
|     patch_get_signal(freqtradebot) |     patch_get_signal(freqtradebot) | ||||||
|  |  | ||||||
|     telegram._profit(update=update, context=MagicMock()) |     telegram._profit(update=update, context=MagicMock()) | ||||||
| @@ -775,10 +668,6 @@ def test_profit_handle(default_conf, update, ticker, ticker_sell_up, fee, | |||||||
|     freqtradebot.enter_positions() |     freqtradebot.enter_positions() | ||||||
|     trade = Trade.query.first() |     trade = Trade.query.first() | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_BUY order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_buy_order, limit_buy_order['symbol'], 'buy') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     context = MagicMock() |     context = MagicMock() | ||||||
|     # Test with invalid 2nd argument (should silently pass) |     # Test with invalid 2nd argument (should silently pass) | ||||||
|     context.args = ["aaa"] |     context.args = ["aaa"] | ||||||
| @@ -786,15 +675,16 @@ def test_profit_handle(default_conf, update, ticker, ticker_sell_up, fee, | |||||||
|     assert msg_mock.call_count == 1 |     assert msg_mock.call_count == 1 | ||||||
|     assert 'No closed trade' in msg_mock.call_args_list[-1][0][0] |     assert 'No closed trade' in msg_mock.call_args_list[-1][0][0] | ||||||
|     assert '*ROI:* All trades' in msg_mock.call_args_list[-1][0][0] |     assert '*ROI:* All trades' in msg_mock.call_args_list[-1][0][0] | ||||||
|     mocker.patch('freqtrade.wallets.Wallets.get_starting_balance', return_value=0.01) |     mocker.patch('freqtrade.wallets.Wallets.get_starting_balance', return_value=1000) | ||||||
|     assert ('∙ `-0.000005 BTC (-0.50%) (-0.0 \N{GREEK CAPITAL LETTER SIGMA}%)`' |     assert ('∙ `0.298 USDT (0.50%) (0.03 \N{GREEK CAPITAL LETTER SIGMA}%)`' | ||||||
|             in msg_mock.call_args_list[-1][0][0]) |             in msg_mock.call_args_list[-1][0][0]) | ||||||
|     msg_mock.reset_mock() |     msg_mock.reset_mock() | ||||||
|  |  | ||||||
|     # Update the ticker with a market going up |     # Update the ticker with a market going up | ||||||
|     mocker.patch('freqtrade.exchange.Exchange.fetch_ticker', ticker_sell_up) |     mocker.patch('freqtrade.exchange.Exchange.fetch_ticker', ticker_sell_up) | ||||||
|     # Simulate fulfilled LIMIT_SELL order for trade |     # Simulate fulfilled LIMIT_SELL order for trade | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_sell_order, limit_sell_order['symbol'], 'sell') |     oobj = Order.parse_from_ccxt_object( | ||||||
|  |         limit_sell_order_usdt, limit_sell_order_usdt['symbol'], 'sell') | ||||||
|     trade.update_trade(oobj) |     trade.update_trade(oobj) | ||||||
|  |  | ||||||
|     trade.close_date = datetime.now(timezone.utc) |     trade.close_date = datetime.now(timezone.utc) | ||||||
| @@ -805,15 +695,15 @@ def test_profit_handle(default_conf, update, ticker, ticker_sell_up, fee, | |||||||
|     telegram._profit(update=update, context=context) |     telegram._profit(update=update, context=context) | ||||||
|     assert msg_mock.call_count == 1 |     assert msg_mock.call_count == 1 | ||||||
|     assert '*ROI:* Closed trades' in msg_mock.call_args_list[-1][0][0] |     assert '*ROI:* Closed trades' in msg_mock.call_args_list[-1][0][0] | ||||||
|     assert ('∙ `0.00006217 BTC (6.20%) (0.62 \N{GREEK CAPITAL LETTER SIGMA}%)`' |     assert ('∙ `5.685 USDT (9.45%) (0.57 \N{GREEK CAPITAL LETTER SIGMA}%)`' | ||||||
|             in msg_mock.call_args_list[-1][0][0]) |             in msg_mock.call_args_list[-1][0][0]) | ||||||
|     assert '∙ `0.933 USD`' in msg_mock.call_args_list[-1][0][0] |     assert '∙ `6.253 USD`' in msg_mock.call_args_list[-1][0][0] | ||||||
|     assert '*ROI:* All trades' in msg_mock.call_args_list[-1][0][0] |     assert '*ROI:* All trades' in msg_mock.call_args_list[-1][0][0] | ||||||
|     assert ('∙ `0.00006217 BTC (6.20%) (0.62 \N{GREEK CAPITAL LETTER SIGMA}%)`' |     assert ('∙ `5.685 USDT (9.45%) (0.57 \N{GREEK CAPITAL LETTER SIGMA}%)`' | ||||||
|             in msg_mock.call_args_list[-1][0][0]) |             in msg_mock.call_args_list[-1][0][0]) | ||||||
|     assert '∙ `0.933 USD`' in msg_mock.call_args_list[-1][0][0] |     assert '∙ `6.253 USD`' in msg_mock.call_args_list[-1][0][0] | ||||||
|  |  | ||||||
|     assert '*Best Performing:* `ETH/BTC: 6.20%`' in msg_mock.call_args_list[-1][0][0] |     assert '*Best Performing:* `ETH/USDT: 9.45%`' in msg_mock.call_args_list[-1][0][0] | ||||||
|  |  | ||||||
|  |  | ||||||
| @pytest.mark.parametrize('is_short', [True, False]) | @pytest.mark.parametrize('is_short', [True, False]) | ||||||
| @@ -1350,71 +1240,43 @@ def test_force_enter_no_pair(default_conf, update, mocker) -> None: | |||||||
|     assert fbuy_mock.call_count == 1 |     assert fbuy_mock.call_count == 1 | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_telegram_performance_handle(default_conf, update, ticker, fee, | def test_telegram_performance_handle(default_conf_usdt, update, ticker, fee, mocker) -> None: | ||||||
|                                      limit_buy_order, limit_sell_order, mocker) -> None: |  | ||||||
|  |  | ||||||
|     mocker.patch.multiple( |     mocker.patch.multiple( | ||||||
|         'freqtrade.exchange.Exchange', |         'freqtrade.exchange.Exchange', | ||||||
|         fetch_ticker=ticker, |         fetch_ticker=ticker, | ||||||
|         get_fee=fee, |         get_fee=fee, | ||||||
|     ) |     ) | ||||||
|     telegram, freqtradebot, msg_mock = get_telegram_testobject(mocker, default_conf) |     telegram, freqtradebot, msg_mock = get_telegram_testobject(mocker, default_conf_usdt) | ||||||
|     patch_get_signal(freqtradebot) |  | ||||||
|  |  | ||||||
|     # Create some test data |     # Create some test data | ||||||
|     freqtradebot.enter_positions() |     create_mock_trades_usdt(fee) | ||||||
|     trade = Trade.query.first() |  | ||||||
|     assert trade |  | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_BUY order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_buy_order, limit_buy_order['symbol'], 'buy') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_SELL order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_sell_order, limit_sell_order['symbol'], 'sell') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     trade.close_date = datetime.utcnow() |  | ||||||
|     trade.is_open = False |  | ||||||
|     telegram._performance(update=update, context=MagicMock()) |     telegram._performance(update=update, context=MagicMock()) | ||||||
|     assert msg_mock.call_count == 1 |     assert msg_mock.call_count == 1 | ||||||
|     assert 'Performance' in msg_mock.call_args_list[0][0][0] |     assert 'Performance' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert '<code>ETH/BTC\t0.00006217 BTC (6.20%) (1)</code>' in msg_mock.call_args_list[0][0][0] |     assert '<code>XRP/USDT\t9.842 USDT (10.00%) (1)</code>' in msg_mock.call_args_list[0][0][0] | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_telegram_entry_tag_performance_handle( | def test_telegram_entry_tag_performance_handle( | ||||||
|         default_conf, update, ticker, fee, limit_buy_order, limit_sell_order, mocker) -> None: |         default_conf_usdt, update, ticker, fee, mocker) -> None: | ||||||
|     mocker.patch.multiple( |     mocker.patch.multiple( | ||||||
|         'freqtrade.exchange.Exchange', |         'freqtrade.exchange.Exchange', | ||||||
|         fetch_ticker=ticker, |         fetch_ticker=ticker, | ||||||
|         get_fee=fee, |         get_fee=fee, | ||||||
|     ) |     ) | ||||||
|     telegram, freqtradebot, msg_mock = get_telegram_testobject(mocker, default_conf) |     telegram, freqtradebot, msg_mock = get_telegram_testobject(mocker, default_conf_usdt) | ||||||
|     patch_get_signal(freqtradebot) |     patch_get_signal(freqtradebot) | ||||||
|  |  | ||||||
|     # Create some test data |     create_mock_trades_usdt(fee) | ||||||
|     freqtradebot.enter_positions() |  | ||||||
|     trade = Trade.query.first() |  | ||||||
|     assert trade |  | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_BUY order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_buy_order, limit_buy_order['symbol'], 'buy') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     trade.enter_tag = "TESTBUY" |  | ||||||
|     # Simulate fulfilled LIMIT_SELL order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_sell_order, limit_sell_order['symbol'], 'sell') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     trade.close_date = datetime.utcnow() |  | ||||||
|     trade.is_open = False |  | ||||||
|     context = MagicMock() |     context = MagicMock() | ||||||
|     telegram._enter_tag_performance(update=update, context=context) |     telegram._enter_tag_performance(update=update, context=context) | ||||||
|     assert msg_mock.call_count == 1 |     assert msg_mock.call_count == 1 | ||||||
|     assert 'Entry Tag Performance' in msg_mock.call_args_list[0][0][0] |     assert 'Entry Tag Performance' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert '<code>TESTBUY\t0.00006217 BTC (6.20%) (1)</code>' in msg_mock.call_args_list[0][0][0] |     assert '<code>TEST1\t3.987 USDT (5.00%) (1)</code>' in msg_mock.call_args_list[0][0][0] | ||||||
|  |  | ||||||
|     context.args = [trade.pair] |     context.args = ['XRP/USDT'] | ||||||
|     telegram._enter_tag_performance(update=update, context=context) |     telegram._enter_tag_performance(update=update, context=context) | ||||||
|     assert msg_mock.call_count == 2 |     assert msg_mock.call_count == 2 | ||||||
|  |  | ||||||
| @@ -1427,37 +1289,24 @@ def test_telegram_entry_tag_performance_handle( | |||||||
|     assert "Error" in msg_mock.call_args_list[0][0][0] |     assert "Error" in msg_mock.call_args_list[0][0][0] | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_telegram_exit_reason_performance_handle(default_conf, update, ticker, fee, | def test_telegram_exit_reason_performance_handle(default_conf_usdt, update, ticker, fee, | ||||||
|                                                  limit_buy_order, limit_sell_order, mocker) -> None: |                                                  mocker) -> None: | ||||||
|     mocker.patch.multiple( |     mocker.patch.multiple( | ||||||
|         'freqtrade.exchange.Exchange', |         'freqtrade.exchange.Exchange', | ||||||
|         fetch_ticker=ticker, |         fetch_ticker=ticker, | ||||||
|         get_fee=fee, |         get_fee=fee, | ||||||
|     ) |     ) | ||||||
|     telegram, freqtradebot, msg_mock = get_telegram_testobject(mocker, default_conf) |     telegram, freqtradebot, msg_mock = get_telegram_testobject(mocker, default_conf_usdt) | ||||||
|     patch_get_signal(freqtradebot) |     patch_get_signal(freqtradebot) | ||||||
|  |  | ||||||
|     # Create some test data |     create_mock_trades_usdt(fee) | ||||||
|     freqtradebot.enter_positions() |  | ||||||
|     trade = Trade.query.first() |  | ||||||
|     assert trade |  | ||||||
|     trade.exit_reason = 'TESTSELL' |  | ||||||
|     # Simulate fulfilled LIMIT_BUY order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_buy_order, limit_buy_order['symbol'], 'buy') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_SELL order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_sell_order, limit_sell_order['symbol'], 'sell') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     trade.close_date = datetime.utcnow() |  | ||||||
|     trade.is_open = False |  | ||||||
|     context = MagicMock() |     context = MagicMock() | ||||||
|     telegram._exit_reason_performance(update=update, context=context) |     telegram._exit_reason_performance(update=update, context=context) | ||||||
|     assert msg_mock.call_count == 1 |     assert msg_mock.call_count == 1 | ||||||
|     assert 'Exit Reason Performance' in msg_mock.call_args_list[0][0][0] |     assert 'Exit Reason Performance' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert '<code>TESTSELL\t0.00006217 BTC (6.20%) (1)</code>' in msg_mock.call_args_list[0][0][0] |     assert '<code>roi\t9.842 USDT (10.00%) (1)</code>' in msg_mock.call_args_list[0][0][0] | ||||||
|     context.args = [trade.pair] |     context.args = ['XRP/USDT'] | ||||||
|  |  | ||||||
|     telegram._exit_reason_performance(update=update, context=context) |     telegram._exit_reason_performance(update=update, context=context) | ||||||
|     assert msg_mock.call_count == 2 |     assert msg_mock.call_count == 2 | ||||||
| @@ -1471,43 +1320,27 @@ def test_telegram_exit_reason_performance_handle(default_conf, update, ticker, f | |||||||
|     assert "Error" in msg_mock.call_args_list[0][0][0] |     assert "Error" in msg_mock.call_args_list[0][0][0] | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_telegram_mix_tag_performance_handle(default_conf, update, ticker, fee, | def test_telegram_mix_tag_performance_handle(default_conf_usdt, update, ticker, fee, | ||||||
|                                              limit_buy_order, limit_sell_order, mocker) -> None: |                                              mocker) -> None: | ||||||
|     mocker.patch.multiple( |     mocker.patch.multiple( | ||||||
|         'freqtrade.exchange.Exchange', |         'freqtrade.exchange.Exchange', | ||||||
|         fetch_ticker=ticker, |         fetch_ticker=ticker, | ||||||
|         get_fee=fee, |         get_fee=fee, | ||||||
|     ) |     ) | ||||||
|     telegram, freqtradebot, msg_mock = get_telegram_testobject(mocker, default_conf) |     telegram, freqtradebot, msg_mock = get_telegram_testobject(mocker, default_conf_usdt) | ||||||
|     patch_get_signal(freqtradebot) |     patch_get_signal(freqtradebot) | ||||||
|  |  | ||||||
|     # Create some test data |     # Create some test data | ||||||
|     freqtradebot.enter_positions() |     create_mock_trades_usdt(fee) | ||||||
|     trade = Trade.query.first() |  | ||||||
|     assert trade |  | ||||||
|  |  | ||||||
|     trade.enter_tag = "TESTBUY" |  | ||||||
|     trade.exit_reason = "TESTSELL" |  | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_BUY order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_buy_order, limit_buy_order['symbol'], 'buy') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     # Simulate fulfilled LIMIT_SELL order for trade |  | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_sell_order, limit_sell_order['symbol'], 'sell') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|  |  | ||||||
|     trade.close_date = datetime.utcnow() |  | ||||||
|     trade.is_open = False |  | ||||||
|  |  | ||||||
|     context = MagicMock() |     context = MagicMock() | ||||||
|     telegram._mix_tag_performance(update=update, context=context) |     telegram._mix_tag_performance(update=update, context=context) | ||||||
|     assert msg_mock.call_count == 1 |     assert msg_mock.call_count == 1 | ||||||
|     assert 'Mix Tag Performance' in msg_mock.call_args_list[0][0][0] |     assert 'Mix Tag Performance' in msg_mock.call_args_list[0][0][0] | ||||||
|     assert ('<code>TESTBUY TESTSELL\t0.00006217 BTC (6.20%) (1)</code>' |     assert ('<code>TEST3 roi\t9.842 USDT (10.00%) (1)</code>' | ||||||
|             in msg_mock.call_args_list[0][0][0]) |             in msg_mock.call_args_list[0][0][0]) | ||||||
|  |  | ||||||
|     context.args = [trade.pair] |     context.args = ['XRP/USDT'] | ||||||
|     telegram._mix_tag_performance(update=update, context=context) |     telegram._mix_tag_performance(update=update, context=context) | ||||||
|     assert msg_mock.call_count == 2 |     assert msg_mock.call_count == 2 | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
| # pragma pylint: disable=missing-docstring, C0103, protected-access | # pragma pylint: disable=missing-docstring, C0103, protected-access | ||||||
|  |  | ||||||
|  | from datetime import datetime, timedelta | ||||||
| from unittest.mock import MagicMock | from unittest.mock import MagicMock | ||||||
|  |  | ||||||
| import pytest | import pytest | ||||||
| @@ -7,6 +8,7 @@ from requests import RequestException | |||||||
|  |  | ||||||
| from freqtrade.enums import ExitType, RPCMessageType | from freqtrade.enums import ExitType, RPCMessageType | ||||||
| from freqtrade.rpc import RPC | from freqtrade.rpc import RPC | ||||||
|  | from freqtrade.rpc.discord import Discord | ||||||
| from freqtrade.rpc.webhook import Webhook | from freqtrade.rpc.webhook import Webhook | ||||||
| from tests.conftest import get_patched_freqtradebot, log_has | from tests.conftest import get_patched_freqtradebot, log_has | ||||||
|  |  | ||||||
| @@ -406,3 +408,42 @@ def test__send_msg_with_raw_format(default_conf, mocker, caplog): | |||||||
|     webhook._send_msg(msg) |     webhook._send_msg(msg) | ||||||
|  |  | ||||||
|     assert post.call_args[1] == {'data': msg['data'], 'headers': {'Content-Type': 'text/plain'}} |     assert post.call_args[1] == {'data': msg['data'], 'headers': {'Content-Type': 'text/plain'}} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_send_msg_discord(default_conf, mocker): | ||||||
|  |  | ||||||
|  |     default_conf["discord"] = { | ||||||
|  |         'enabled': True, | ||||||
|  |         'webhook_url': "https://webhookurl..." | ||||||
|  |     } | ||||||
|  |     msg_mock = MagicMock() | ||||||
|  |     mocker.patch("freqtrade.rpc.webhook.Webhook._send_msg", msg_mock) | ||||||
|  |     discord = Discord(RPC(get_patched_freqtradebot(mocker, default_conf)), default_conf) | ||||||
|  |  | ||||||
|  |     msg = { | ||||||
|  |         'type': RPCMessageType.EXIT_FILL, | ||||||
|  |         'trade_id': 1, | ||||||
|  |         'exchange': 'Binance', | ||||||
|  |         'pair': 'ETH/BTC', | ||||||
|  |         'direction': 'Long', | ||||||
|  |         'gain': "profit", | ||||||
|  |         'close_rate': 0.005, | ||||||
|  |         'amount': 0.8, | ||||||
|  |         'order_type': 'limit', | ||||||
|  |         'open_date': datetime.now() - timedelta(days=1), | ||||||
|  |         'close_date': datetime.now(), | ||||||
|  |         'open_rate': 0.004, | ||||||
|  |         'current_rate': 0.005, | ||||||
|  |         'profit_amount': 0.001, | ||||||
|  |         'profit_ratio': 0.20, | ||||||
|  |         'stake_currency': 'BTC', | ||||||
|  |         'enter_tag': 'enter_tagggg', | ||||||
|  |         'exit_reason': ExitType.STOP_LOSS.value, | ||||||
|  |     } | ||||||
|  |     discord.send_msg(msg=msg) | ||||||
|  |  | ||||||
|  |     assert msg_mock.call_count == 1 | ||||||
|  |     assert 'embeds' in msg_mock.call_args_list[0][0][0] | ||||||
|  |     assert 'title' in msg_mock.call_args_list[0][0][0]['embeds'][0] | ||||||
|  |     assert 'color' in msg_mock.call_args_list[0][0][0]['embeds'][0] | ||||||
|  |     assert 'fields' in msg_mock.call_args_list[0][0][0]['embeds'][0] | ||||||
|   | |||||||
| @@ -178,8 +178,8 @@ class StrategyTestV3(IStrategy): | |||||||
|         return dataframe |         return dataframe | ||||||
|  |  | ||||||
|     def leverage(self, pair: str, current_time: datetime, current_rate: float, |     def leverage(self, pair: str, current_time: datetime, current_rate: float, | ||||||
|                  proposed_leverage: float, max_leverage: float, side: str, |                  proposed_leverage: float, max_leverage: float, entry_tag: Optional[str], | ||||||
|                  **kwargs) -> float: |                  side: str, **kwargs) -> float: | ||||||
|         # Return 3.0 in all cases. |         # Return 3.0 in all cases. | ||||||
|         # Bot-logic must make sure it's an allowed leverage and eventually adjust accordingly. |         # Bot-logic must make sure it's an allowed leverage and eventually adjust accordingly. | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										175
									
								
								tests/strategy/strats/strategy_test_v3_analysis.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										175
									
								
								tests/strategy/strats/strategy_test_v3_analysis.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,175 @@ | |||||||
|  | # pragma pylint: disable=missing-docstring, invalid-name, pointless-string-statement | ||||||
|  |  | ||||||
|  | import talib.abstract as ta | ||||||
|  | from pandas import DataFrame | ||||||
|  |  | ||||||
|  | import freqtrade.vendor.qtpylib.indicators as qtpylib | ||||||
|  | from freqtrade.strategy import (BooleanParameter, DecimalParameter, IntParameter, IStrategy, | ||||||
|  |                                 RealParameter) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class StrategyTestV3Analysis(IStrategy): | ||||||
|  |     """ | ||||||
|  |     Strategy used by tests freqtrade bot. | ||||||
|  |     Please do not modify this strategy, it's  intended for internal use only. | ||||||
|  |     Please look at the SampleStrategy in the user_data/strategy directory | ||||||
|  |     or strategy repository https://github.com/freqtrade/freqtrade-strategies | ||||||
|  |     for samples and inspiration. | ||||||
|  |     """ | ||||||
|  |     INTERFACE_VERSION = 3 | ||||||
|  |  | ||||||
|  |     # Minimal ROI designed for the strategy | ||||||
|  |     minimal_roi = { | ||||||
|  |         "40": 0.0, | ||||||
|  |         "30": 0.01, | ||||||
|  |         "20": 0.02, | ||||||
|  |         "0": 0.04 | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     # Optimal stoploss designed for the strategy | ||||||
|  |     stoploss = -0.10 | ||||||
|  |  | ||||||
|  |     # Optimal timeframe for the strategy | ||||||
|  |     timeframe = '5m' | ||||||
|  |  | ||||||
|  |     # Optional order type mapping | ||||||
|  |     order_types = { | ||||||
|  |         'entry': 'limit', | ||||||
|  |         'exit': 'limit', | ||||||
|  |         'stoploss': 'limit', | ||||||
|  |         'stoploss_on_exchange': False | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     # Number of candles the strategy requires before producing valid signals | ||||||
|  |     startup_candle_count: int = 20 | ||||||
|  |  | ||||||
|  |     # Optional time in force for orders | ||||||
|  |     order_time_in_force = { | ||||||
|  |         'entry': 'gtc', | ||||||
|  |         'exit': 'gtc', | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     buy_params = { | ||||||
|  |         'buy_rsi': 35, | ||||||
|  |         # Intentionally not specified, so "default" is tested | ||||||
|  |         # 'buy_plusdi': 0.4 | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     sell_params = { | ||||||
|  |         'sell_rsi': 74, | ||||||
|  |         'sell_minusdi': 0.4 | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     buy_rsi = IntParameter([0, 50], default=30, space='buy') | ||||||
|  |     buy_plusdi = RealParameter(low=0, high=1, default=0.5, space='buy') | ||||||
|  |     sell_rsi = IntParameter(low=50, high=100, default=70, space='sell') | ||||||
|  |     sell_minusdi = DecimalParameter(low=0, high=1, default=0.5001, decimals=3, space='sell', | ||||||
|  |                                     load=False) | ||||||
|  |     protection_enabled = BooleanParameter(default=True) | ||||||
|  |     protection_cooldown_lookback = IntParameter([0, 50], default=30) | ||||||
|  |  | ||||||
|  |     # TODO: Can this work with protection tests? (replace HyperoptableStrategy implicitly ... ) | ||||||
|  |     # @property | ||||||
|  |     # def protections(self): | ||||||
|  |     #     prot = [] | ||||||
|  |     #     if self.protection_enabled.value: | ||||||
|  |     #         prot.append({ | ||||||
|  |     #             "method": "CooldownPeriod", | ||||||
|  |     #             "stop_duration_candles": self.protection_cooldown_lookback.value | ||||||
|  |     #         }) | ||||||
|  |     #     return prot | ||||||
|  |  | ||||||
|  |     bot_started = False | ||||||
|  |  | ||||||
|  |     def bot_start(self): | ||||||
|  |         self.bot_started = True | ||||||
|  |  | ||||||
|  |     def informative_pairs(self): | ||||||
|  |  | ||||||
|  |         return [] | ||||||
|  |  | ||||||
|  |     def populate_indicators(self, dataframe: DataFrame, metadata: dict) -> DataFrame: | ||||||
|  |  | ||||||
|  |         # Momentum Indicator | ||||||
|  |         # ------------------------------------ | ||||||
|  |  | ||||||
|  |         # ADX | ||||||
|  |         dataframe['adx'] = ta.ADX(dataframe) | ||||||
|  |  | ||||||
|  |         # MACD | ||||||
|  |         macd = ta.MACD(dataframe) | ||||||
|  |         dataframe['macd'] = macd['macd'] | ||||||
|  |         dataframe['macdsignal'] = macd['macdsignal'] | ||||||
|  |         dataframe['macdhist'] = macd['macdhist'] | ||||||
|  |  | ||||||
|  |         # Minus Directional Indicator / Movement | ||||||
|  |         dataframe['minus_di'] = ta.MINUS_DI(dataframe) | ||||||
|  |  | ||||||
|  |         # Plus Directional Indicator / Movement | ||||||
|  |         dataframe['plus_di'] = ta.PLUS_DI(dataframe) | ||||||
|  |  | ||||||
|  |         # RSI | ||||||
|  |         dataframe['rsi'] = ta.RSI(dataframe) | ||||||
|  |  | ||||||
|  |         # Stoch fast | ||||||
|  |         stoch_fast = ta.STOCHF(dataframe) | ||||||
|  |         dataframe['fastd'] = stoch_fast['fastd'] | ||||||
|  |         dataframe['fastk'] = stoch_fast['fastk'] | ||||||
|  |  | ||||||
|  |         # Bollinger bands | ||||||
|  |         bollinger = qtpylib.bollinger_bands(qtpylib.typical_price(dataframe), window=20, stds=2) | ||||||
|  |         dataframe['bb_lowerband'] = bollinger['lower'] | ||||||
|  |         dataframe['bb_middleband'] = bollinger['mid'] | ||||||
|  |         dataframe['bb_upperband'] = bollinger['upper'] | ||||||
|  |  | ||||||
|  |         # EMA - Exponential Moving Average | ||||||
|  |         dataframe['ema10'] = ta.EMA(dataframe, timeperiod=10) | ||||||
|  |  | ||||||
|  |         return dataframe | ||||||
|  |  | ||||||
|  |     def populate_entry_trend(self, dataframe: DataFrame, metadata: dict) -> DataFrame: | ||||||
|  |  | ||||||
|  |         dataframe.loc[ | ||||||
|  |             ( | ||||||
|  |                 (dataframe['rsi'] < self.buy_rsi.value) & | ||||||
|  |                 (dataframe['fastd'] < 35) & | ||||||
|  |                 (dataframe['adx'] > 30) & | ||||||
|  |                 (dataframe['plus_di'] > self.buy_plusdi.value) | ||||||
|  |             ) | | ||||||
|  |             ( | ||||||
|  |                 (dataframe['adx'] > 65) & | ||||||
|  |                 (dataframe['plus_di'] > self.buy_plusdi.value) | ||||||
|  |             ), | ||||||
|  |             ['enter_long', 'enter_tag']] = 1, 'enter_tag_long' | ||||||
|  |  | ||||||
|  |         dataframe.loc[ | ||||||
|  |             ( | ||||||
|  |                 qtpylib.crossed_below(dataframe['rsi'], self.sell_rsi.value) | ||||||
|  |             ), | ||||||
|  |             ['enter_short', 'enter_tag']] = 1, 'enter_tag_short' | ||||||
|  |  | ||||||
|  |         return dataframe | ||||||
|  |  | ||||||
|  |     def populate_exit_trend(self, dataframe: DataFrame, metadata: dict) -> DataFrame: | ||||||
|  |         dataframe.loc[ | ||||||
|  |             ( | ||||||
|  |                 ( | ||||||
|  |                     (qtpylib.crossed_above(dataframe['rsi'], self.sell_rsi.value)) | | ||||||
|  |                     (qtpylib.crossed_above(dataframe['fastd'], 70)) | ||||||
|  |                 ) & | ||||||
|  |                 (dataframe['adx'] > 10) & | ||||||
|  |                 (dataframe['minus_di'] > 0) | ||||||
|  |             ) | | ||||||
|  |             ( | ||||||
|  |                 (dataframe['adx'] > 70) & | ||||||
|  |                 (dataframe['minus_di'] > self.sell_minusdi.value) | ||||||
|  |             ), | ||||||
|  |             ['exit_long', 'exit_tag']] = 1, 'exit_tag_long' | ||||||
|  |  | ||||||
|  |         dataframe.loc[ | ||||||
|  |             ( | ||||||
|  |                 qtpylib.crossed_above(dataframe['rsi'], self.buy_rsi.value) | ||||||
|  |             ), | ||||||
|  |             ['exit_long', 'exit_tag']] = 1, 'exit_tag_short' | ||||||
|  |  | ||||||
|  |         return dataframe | ||||||
| @@ -20,7 +20,8 @@ from freqtrade.strategy.hyper import detect_parameters | |||||||
| from freqtrade.strategy.parameters import (BaseParameter, BooleanParameter, CategoricalParameter, | from freqtrade.strategy.parameters import (BaseParameter, BooleanParameter, CategoricalParameter, | ||||||
|                                            DecimalParameter, IntParameter, RealParameter) |                                            DecimalParameter, IntParameter, RealParameter) | ||||||
| from freqtrade.strategy.strategy_wrapper import strategy_safe_wrapper | from freqtrade.strategy.strategy_wrapper import strategy_safe_wrapper | ||||||
| from tests.conftest import CURRENT_TEST_STRATEGY, TRADE_SIDES, log_has, log_has_re | from tests.conftest import (CURRENT_TEST_STRATEGY, TRADE_SIDES, create_mock_trades, log_has, | ||||||
|  |                             log_has_re) | ||||||
|  |  | ||||||
| from .strats.strategy_test_v3 import StrategyTestV3 | from .strats.strategy_test_v3 import StrategyTestV3 | ||||||
|  |  | ||||||
| @@ -615,6 +616,7 @@ def test_leverage_callback(default_conf, side) -> None: | |||||||
|         proposed_leverage=1.0, |         proposed_leverage=1.0, | ||||||
|         max_leverage=5.0, |         max_leverage=5.0, | ||||||
|         side=side, |         side=side, | ||||||
|  |         entry_tag=None, | ||||||
|         ) == 1 |         ) == 1 | ||||||
|  |  | ||||||
|     default_conf['strategy'] = CURRENT_TEST_STRATEGY |     default_conf['strategy'] = CURRENT_TEST_STRATEGY | ||||||
| @@ -626,6 +628,7 @@ def test_leverage_callback(default_conf, side) -> None: | |||||||
|         proposed_leverage=1.0, |         proposed_leverage=1.0, | ||||||
|         max_leverage=5.0, |         max_leverage=5.0, | ||||||
|         side=side, |         side=side, | ||||||
|  |         entry_tag='entry_tag_test', | ||||||
|         ) == 3 |         ) == 3 | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -810,6 +813,28 @@ def test_strategy_safe_wrapper(value): | |||||||
|     assert ret == value |     assert ret == value | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.usefixtures("init_persistence") | ||||||
|  | def test_strategy_safe_wrapper_trade_copy(fee): | ||||||
|  |     create_mock_trades(fee) | ||||||
|  |  | ||||||
|  |     def working_method(trade): | ||||||
|  |         assert len(trade.orders) > 0 | ||||||
|  |         assert trade.orders | ||||||
|  |         trade.orders = [] | ||||||
|  |         assert len(trade.orders) == 0 | ||||||
|  |         return trade | ||||||
|  |  | ||||||
|  |     trade = Trade.get_open_trades()[0] | ||||||
|  |     # Don't assert anything before strategy_wrapper. | ||||||
|  |     # This ensures that relationship loading works correctly. | ||||||
|  |     ret = strategy_safe_wrapper(working_method, message='DeadBeef')(trade=trade) | ||||||
|  |     assert isinstance(ret, Trade) | ||||||
|  |     assert id(trade) != id(ret) | ||||||
|  |     # Did not modify the original order | ||||||
|  |     assert len(trade.orders) > 0 | ||||||
|  |     assert len(ret.orders) == 0 | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_hyperopt_parameters(): | def test_hyperopt_parameters(): | ||||||
|     from skopt.space import Categorical, Integer, Real |     from skopt.space import Categorical, Integer, Real | ||||||
|     with pytest.raises(OperationalException, match=r"Name is determined.*"): |     with pytest.raises(OperationalException, match=r"Name is determined.*"): | ||||||
|   | |||||||
| @@ -34,7 +34,7 @@ def test_search_all_strategies_no_failed(): | |||||||
|     directory = Path(__file__).parent / "strats" |     directory = Path(__file__).parent / "strats" | ||||||
|     strategies = StrategyResolver.search_all_objects(directory, enum_failed=False) |     strategies = StrategyResolver.search_all_objects(directory, enum_failed=False) | ||||||
|     assert isinstance(strategies, list) |     assert isinstance(strategies, list) | ||||||
|     assert len(strategies) == 5 |     assert len(strategies) == 6 | ||||||
|     assert isinstance(strategies[0], dict) |     assert isinstance(strategies[0], dict) | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -42,10 +42,10 @@ def test_search_all_strategies_with_failed(): | |||||||
|     directory = Path(__file__).parent / "strats" |     directory = Path(__file__).parent / "strats" | ||||||
|     strategies = StrategyResolver.search_all_objects(directory, enum_failed=True) |     strategies = StrategyResolver.search_all_objects(directory, enum_failed=True) | ||||||
|     assert isinstance(strategies, list) |     assert isinstance(strategies, list) | ||||||
|     assert len(strategies) == 6 |     assert len(strategies) == 7 | ||||||
|     # with enum_failed=True search_all_objects() shall find 2 good strategies |     # with enum_failed=True search_all_objects() shall find 2 good strategies | ||||||
|     # and 1 which fails to load |     # and 1 which fails to load | ||||||
|     assert len([x for x in strategies if x['class'] is not None]) == 5 |     assert len([x for x in strategies if x['class'] is not None]) == 6 | ||||||
|     assert len([x for x in strategies if x['class'] is None]) == 1 |     assert len([x for x in strategies if x['class'] is None]) == 1 | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -210,13 +210,14 @@ def test_edge_overrides_stoploss(limit_order, fee, caplog, mocker, | |||||||
|     # |     # | ||||||
|     # mocking the ticker: price is falling ... |     # mocking the ticker: price is falling ... | ||||||
|     enter_price = limit_order['buy']['price'] |     enter_price = limit_order['buy']['price'] | ||||||
|  |     ticker_val = { | ||||||
|  |             'bid': enter_price, | ||||||
|  |             'ask': enter_price, | ||||||
|  |             'last': enter_price, | ||||||
|  |         } | ||||||
|     mocker.patch.multiple( |     mocker.patch.multiple( | ||||||
|         'freqtrade.exchange.Exchange', |         'freqtrade.exchange.Exchange', | ||||||
|         fetch_ticker=MagicMock(return_value={ |         fetch_ticker=MagicMock(return_value=ticker_val), | ||||||
|             'bid': enter_price * buy_price_mult, |  | ||||||
|             'ask': enter_price * buy_price_mult, |  | ||||||
|             'last': enter_price * buy_price_mult, |  | ||||||
|         }), |  | ||||||
|         get_fee=fee, |         get_fee=fee, | ||||||
|     ) |     ) | ||||||
|     ############################################# |     ############################################# | ||||||
| @@ -229,9 +230,12 @@ def test_edge_overrides_stoploss(limit_order, fee, caplog, mocker, | |||||||
|     freqtrade.enter_positions() |     freqtrade.enter_positions() | ||||||
|     trade = Trade.query.first() |     trade = Trade.query.first() | ||||||
|     caplog.clear() |     caplog.clear() | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_order['buy'], 'ADA/USDT', 'buy') |  | ||||||
|     trade.update_trade(oobj) |  | ||||||
|     ############################################# |     ############################################# | ||||||
|  |     ticker_val.update({ | ||||||
|  |             'bid': enter_price * buy_price_mult, | ||||||
|  |             'ask': enter_price * buy_price_mult, | ||||||
|  |             'last': enter_price * buy_price_mult, | ||||||
|  |     }) | ||||||
|  |  | ||||||
|     # stoploss shoud be hit |     # stoploss shoud be hit | ||||||
|     assert freqtrade.handle_trade(trade) is not ignore_strat_sl |     assert freqtrade.handle_trade(trade) is not ignore_strat_sl | ||||||
| @@ -3771,6 +3775,7 @@ def test_exit_profit_only( | |||||||
|     trade = Trade.query.first() |     trade = Trade.query.first() | ||||||
|     assert trade.is_short == is_short |     assert trade.is_short == is_short | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_order[eside], limit_order[eside]['symbol'], eside) |     oobj = Order.parse_from_ccxt_object(limit_order[eside], limit_order[eside]['symbol'], eside) | ||||||
|  |     trade.update_order(limit_order[eside]) | ||||||
|     trade.update_trade(oobj) |     trade.update_trade(oobj) | ||||||
|     freqtrade.wallets.update() |     freqtrade.wallets.update() | ||||||
|     if profit_only: |     if profit_only: | ||||||
| @@ -4059,6 +4064,7 @@ def test_trailing_stop_loss_positive( | |||||||
|     trade = Trade.query.first() |     trade = Trade.query.first() | ||||||
|     assert trade.is_short == is_short |     assert trade.is_short == is_short | ||||||
|     oobj = Order.parse_from_ccxt_object(limit_order[eside], limit_order[eside]['symbol'], eside) |     oobj = Order.parse_from_ccxt_object(limit_order[eside], limit_order[eside]['symbol'], eside) | ||||||
|  |     trade.update_order(limit_order[eside]) | ||||||
|     trade.update_trade(oobj) |     trade.update_trade(oobj) | ||||||
|     caplog.set_level(logging.DEBUG) |     caplog.set_level(logging.DEBUG) | ||||||
|     # stop-loss not reached |     # stop-loss not reached | ||||||
| @@ -4802,10 +4808,19 @@ def test_startup_update_open_orders(mocker, default_conf_usdt, fee, caplog, is_s | |||||||
|     assert len(Order.get_open_orders()) == 2 |     assert len(Order.get_open_orders()) == 2 | ||||||
|  |  | ||||||
|     caplog.clear() |     caplog.clear() | ||||||
|     mocker.patch('freqtrade.exchange.Exchange.fetch_order', side_effect=InvalidOrderException) |     mocker.patch('freqtrade.exchange.Exchange.fetch_order', side_effect=ExchangeError) | ||||||
|     freqtrade.startup_update_open_orders() |     freqtrade.startup_update_open_orders() | ||||||
|     assert log_has_re(r"Error updating Order .*", caplog) |     assert log_has_re(r"Error updating Order .*", caplog) | ||||||
|  |  | ||||||
|  |     mocker.patch('freqtrade.exchange.Exchange.fetch_order', side_effect=InvalidOrderException) | ||||||
|  |     hto_mock = mocker.patch('freqtrade.freqtradebot.FreqtradeBot.handle_timedout_order') | ||||||
|  |     # Orders which are no longer found after X days should be assumed as canceled. | ||||||
|  |     freqtrade.startup_update_open_orders() | ||||||
|  |     assert log_has_re(r"Order is older than \d days.*", caplog) | ||||||
|  |     assert hto_mock.call_count == 2 | ||||||
|  |     assert hto_mock.call_args_list[0][0][0]['status'] == 'canceled' | ||||||
|  |     assert hto_mock.call_args_list[1][0][0]['status'] == 'canceled' | ||||||
|  |  | ||||||
|  |  | ||||||
| @pytest.mark.usefixtures("init_persistence") | @pytest.mark.usefixtures("init_persistence") | ||||||
| @pytest.mark.parametrize("is_short", [False, True]) | @pytest.mark.parametrize("is_short", [False, True]) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user