Skip to content

Commit 02fc59d

Browse files
committed
Merge branch 'develop' into backtest_live_models
2 parents 4e1bf79 + c6d2eed commit 02fc59d

File tree

5 files changed

+54
-22
lines changed

5 files changed

+54
-22
lines changed

docs/freqai-configuration.md

+32-5
Original file line numberDiff line numberDiff line change
@@ -204,17 +204,44 @@ If this value is set, FreqAI will initially use the predictions from the trainin
204204

205205
## Using different prediction models
206206

207-
FreqAI has multiple example prediction model libraries that are ready to be used as is via the flag `--freqaimodel`. These libraries include `Catboost`, `LightGBM`, and `XGBoost` regression, classification, and multi-target models, and can be found in `freqai/prediction_models/`. However, it is possible to customize and create your own prediction models using the `IFreqaiModel` class. You are encouraged to inherit `fit()`, `train()`, and `predict()` to let these customize various aspects of the training procedures.
207+
FreqAI has multiple example prediction model libraries that are ready to be used as is via the flag `--freqaimodel`. These libraries include `CatBoost`, `LightGBM`, and `XGBoost` regression, classification, and multi-target models, and can be found in `freqai/prediction_models/`.
208208

209-
You can place custom FreqAI models in `user_data/freqaimodels` - and freqtrade will pick them up from there based on the provided `--freqaimodel` name - which has to correspond to the class name of your custom model.
209+
Regression and classification models differ in what targets they predict - a regression model will predict a target of continuous values, for example what price BTC will be at tomorrow, whilst a classifier will predict a target of discrete values, for example if the price of BTC will go up tomorrow or not. This means that you have to specify your targets differently depending on which model type you are using (see details [below](#setting-model-targets)).
210+
211+
All of the aforementioned model libraries implement gradient boosted decision tree algorithms. They all work on the principle of ensemble learning, where predictions from multiple simple learners are combined to get a final prediction that is more stable and generalized. The simple learners in this case are decision trees. Gradient boosting refers to the method of learning, where each simple learner is built in sequence - the subsequent learner is used to improve on the error from the previous learner. If you want to learn more about the different model libraries you can find the information in their respective docs:
212+
213+
* CatBoost: https://catboost.ai/en/docs/
214+
* LightGBM: https://lightgbm.readthedocs.io/en/v3.3.2/#
215+
* XGBoost: https://xgboost.readthedocs.io/en/stable/#
216+
217+
There are also numerous online articles describing and comparing the algorithms. Some relatively light-weight examples would be [CatBoost vs. LightGBM vs. XGBoost — Which is the best algorithm?](https://towardsdatascience.com/catboost-vs-lightgbm-vs-xgboost-c80f40662924#:~:text=In%20CatBoost%2C%20symmetric%20trees%2C%20or,the%20same%20depth%20can%20differ.) and [XGBoost, LightGBM or CatBoost — which boosting algorithm should I use?](https://medium.com/riskified-technology/xgboost-lightgbm-or-catboost-which-boosting-algorithm-should-i-use-e7fda7bb36bc). Keep in mind that the performance of each model is highly dependent on the application and so any reported metrics might not be true for your particular use of the model.
218+
219+
Apart from the models already available in FreqAI, it is also possible to customize and create your own prediction models using the `IFreqaiModel` class. You are encouraged to inherit `fit()`, `train()`, and `predict()` to customize various aspects of the training procedures. You can place custom FreqAI models in `user_data/freqaimodels` - and freqtrade will pick them up from there based on the provided `--freqaimodel` name - which has to correspond to the class name of your custom model.
210220
Make sure to use unique names to avoid overriding built-in models.
211221

212-
### Setting classifier targets
222+
### Setting model targets
213223

214-
FreqAI includes a variety of classifiers, such as the `CatboostClassifier` via the flag `--freqaimodel CatboostClassifier`. If you elects to use a classifier, the classes need to be set using strings. For example:
224+
#### Regressors
225+
226+
If you are using a regressor, you need to specify a target that has continuous values. FreqAI includes a variety of regressors, such as the `CatboostRegressor`via the flag `--freqaimodel CatboostRegressor`. An example of how you could set a regression target for predicting the price 100 candles into the future would be
227+
228+
```python
229+
df['&s-close_price'] = df['close'].shift(-100)
230+
```
231+
232+
If you want to predict multiple targets, you need to define multiple labels using the same syntax as shown above.
233+
234+
#### Classifiers
235+
236+
If you are using a classifier, you need to specify a target that has discrete values. FreqAI includes a variety of classifiers, such as the `CatboostClassifier` via the flag `--freqaimodel CatboostClassifier`. If you elects to use a classifier, the classes need to be set using strings. For example, if you want to predict if the price 100 candles into the future goes up or down you would set
215237

216238
```python
217239
df['&s-up_or_down'] = np.where( df["close"].shift(-100) > df["close"], 'up', 'down')
218240
```
219241

220-
Additionally, the example classifier models do not accommodate multiple labels, but they do allow multi-class classification within a single label column.
242+
If you want to predict multiple targets you must specify all labels in the same label column. You could, for example, add the label `same` to define where the price was unchanged by setting
243+
244+
```python
245+
df['&s-up_or_down'] = np.where( df["close"].shift(-100) > df["close"], 'up', 'down')
246+
df['&s-up_or_down'] = np.where( df["close"].shift(-100) == df["close"], 'same', df['&s-up_or_down'])
247+
```

docs/freqai-parameter-table.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -42,7 +42,7 @@ Mandatory parameters are marked as **Required** and have to be set in one of the
4242
| `test_size` | The fraction of data that should be used for testing instead of training. <br> **Datatype:** Positive float < 1.
4343
| `shuffle` | Shuffle the training data points during training. Typically, to not remove the chronological order of data in time-series forecasting, this is set to `False`. <br> **Datatype:** Boolean. <br> Defaut: `False`.
4444
| | **Model training parameters**
45-
| `model_training_parameters` | A flexible dictionary that includes all parameters available by the selected model library. For example, if you use `LightGBMRegressor`, this dictionary can contain any parameter available by the `LightGBMRegressor` [here](https://lightgbm.readthedocs.io/en/latest/pythonapi/lightgbm.LGBMRegressor.html) (external website). If you select a different model, this dictionary can contain any parameter from that model. <br> **Datatype:** Dictionary.
45+
| `model_training_parameters` | A flexible dictionary that includes all parameters available by the selected model library. For example, if you use `LightGBMRegressor`, this dictionary can contain any parameter available by the `LightGBMRegressor` [here](https://lightgbm.readthedocs.io/en/latest/pythonapi/lightgbm.LGBMRegressor.html) (external website). If you select a different model, this dictionary can contain any parameter from that model. A list of the currently available models can be found [here](freqai-configuration.md#using-different-prediction-models). <br> **Datatype:** Dictionary.
4646
| `n_estimators` | The number of boosted trees to fit in the training of the model. <br> **Datatype:** Integer.
4747
| `learning_rate` | Boosting learning rate during training of the model. <br> **Datatype:** Float.
4848
| `n_jobs`, `thread_count`, `task_type` | Set the number of threads for parallel processing and the `task_type` (`gpu` or `cpu`). Different model libraries use different parameter names. <br> **Datatype:** Float.

docs/freqai-running.md

+2-3
Original file line numberDiff line numberDiff line change
@@ -156,18 +156,17 @@ This specific hyperopt would help you understand the appropriate `DI_values` for
156156

157157
## Using Tensorboard
158158

159-
Catboost models benefit from tracking training metrics via Tensorboard. You can take advantage of the FreqAI integration to track training and evaluation performance across all coins and across all retrainings. Tensorboard is activated via the following command:
159+
CatBoost models benefit from tracking training metrics via Tensorboard. You can take advantage of the FreqAI integration to track training and evaluation performance across all coins and across all retrainings. Tensorboard is activated via the following command:
160160

161161
```bash
162162
cd freqtrade
163163
tensorboard --logdir user_data/models/unique-id
164164
```
165165

166-
where `unique-id` is the `identifier` set in the `freqai` configuration file. This command must be run in a separate shell if the user wishes to view the output in their browser at 127.0.0.1:6060 (6060 is the default port used by Tensorboard).
166+
where `unique-id` is the `identifier` set in the `freqai` configuration file. This command must be run in a separate shell if you wish to view the output in your browser at 127.0.0.1:6060 (6060 is the default port used by Tensorboard).
167167

168168
![tensorboard](assets/tensorboard.jpg)
169169

170-
171170
## Setting up a follower
172171

173172
You can indicate to the bot that it should not train models, but instead should look for models trained by a leader with a specific `identifier` by defining:

freqtrade/optimize/backtesting.py

+17-12
Original file line numberDiff line numberDiff line change
@@ -621,13 +621,16 @@ def _get_exit_for_signal(
621621
exit_reason = row[EXIT_TAG_IDX]
622622
# Custom exit pricing only for exit-signals
623623
if order_type == 'limit':
624-
close_rate = strategy_safe_wrapper(self.strategy.custom_exit_price,
625-
default_retval=close_rate)(
624+
rate = strategy_safe_wrapper(self.strategy.custom_exit_price,
625+
default_retval=close_rate)(
626626
pair=trade.pair,
627627
trade=trade, # type: ignore[arg-type]
628628
current_time=exit_candle_time,
629629
proposed_rate=close_rate, current_profit=current_profit,
630630
exit_tag=exit_reason)
631+
if rate != close_rate:
632+
close_rate = price_to_precision(rate, trade.price_precision,
633+
self.precision_mode)
631634
# We can't place orders lower than current low.
632635
# freqtrade does not support this in live, and the order would fill immediately
633636
if trade.is_short:
@@ -664,7 +667,6 @@ def _exit_trade(self, trade: LocalTrade, sell_row: Tuple,
664667
# amount = amount or trade.amount
665668
amount = amount_to_contract_precision(amount or trade.amount, trade.amount_precision,
666669
self.precision_mode, trade.contract_size)
667-
rate = price_to_precision(close_rate, trade.price_precision, self.precision_mode)
668670
order = Order(
669671
id=self.order_id_counter,
670672
ft_trade_id=trade.id,
@@ -678,12 +680,12 @@ def _exit_trade(self, trade: LocalTrade, sell_row: Tuple,
678680
side=trade.exit_side,
679681
order_type=order_type,
680682
status="open",
681-
price=rate,
682-
average=rate,
683+
price=close_rate,
684+
average=close_rate,
683685
amount=amount,
684686
filled=0,
685687
remaining=amount,
686-
cost=amount * rate,
688+
cost=amount * close_rate,
687689
)
688690
trade.orders.append(order)
689691
return trade
@@ -730,18 +732,21 @@ def _get_exit_trade_entry(self, trade: LocalTrade, row: Tuple) -> Optional[Local
730732
def get_valid_price_and_stake(
731733
self, pair: str, row: Tuple, propose_rate: float, stake_amount: float,
732734
direction: LongShort, current_time: datetime, entry_tag: Optional[str],
733-
trade: Optional[LocalTrade], order_type: str
735+
trade: Optional[LocalTrade], order_type: str, price_precision: Optional[float]
734736
) -> Tuple[float, float, float, float]:
735737

736738
if order_type == 'limit':
737-
propose_rate = strategy_safe_wrapper(self.strategy.custom_entry_price,
738-
default_retval=propose_rate)(
739+
new_rate = strategy_safe_wrapper(self.strategy.custom_entry_price,
740+
default_retval=propose_rate)(
739741
pair=pair, current_time=current_time,
740742
proposed_rate=propose_rate, entry_tag=entry_tag,
741743
side=direction,
742744
) # default value is the open rate
743745
# We can't place orders higher than current high (otherwise it'd be a stop limit entry)
744746
# which freqtrade does not support in live.
747+
if new_rate != propose_rate:
748+
propose_rate = price_to_precision(new_rate, price_precision,
749+
self.precision_mode)
745750
if direction == "short":
746751
propose_rate = max(propose_rate, row[LOW_IDX])
747752
else:
@@ -803,9 +808,11 @@ def _enter_trade(self, pair: str, row: Tuple, direction: LongShort,
803808
pos_adjust = trade is not None and requested_rate is None
804809

805810
stake_amount_ = stake_amount or (trade.stake_amount if trade else 0.0)
811+
precision_price = self.exchange.get_precision_price(pair)
812+
806813
propose_rate, stake_amount, leverage, min_stake_amount = self.get_valid_price_and_stake(
807814
pair, row, row[OPEN_IDX], stake_amount_, direction, current_time, entry_tag, trade,
808-
order_type
815+
order_type, precision_price,
809816
)
810817

811818
# replace proposed rate if another rate was requested
@@ -821,8 +828,6 @@ def _enter_trade(self, pair: str, row: Tuple, direction: LongShort,
821828
if stake_amount and (not min_stake_amount or stake_amount > min_stake_amount):
822829
self.order_id_counter += 1
823830
base_currency = self.exchange.get_pair_base_currency(pair)
824-
precision_price = self.exchange.get_precision_price(pair)
825-
propose_rate = price_to_precision(propose_rate, precision_price, self.precision_mode)
826831
amount_p = (stake_amount / propose_rate) * leverage
827832

828833
contract_size = self.exchange.get_contract_size(pair)

tests/optimize/test_hyperopt.py

+2-1
Original file line numberDiff line numberDiff line change
@@ -910,8 +910,9 @@ def test_in_strategy_auto_hyperopt_with_parallel(mocker, hyperopt_conf, tmpdir,
910910
})
911911
hyperopt = Hyperopt(hyperopt_conf)
912912
hyperopt.backtesting.exchange.get_max_leverage = lambda *x, **xx: 1.0
913-
hyperopt.backtesting.exchange.get_min_pair_stake_amount = lambda *x, **xx: 1.0
913+
hyperopt.backtesting.exchange.get_min_pair_stake_amount = lambda *x, **xx: 0.00001
914914
hyperopt.backtesting.exchange.get_max_pair_stake_amount = lambda *x, **xx: 100.0
915+
hyperopt.backtesting.exchange._markets = get_markets()
915916

916917
assert isinstance(hyperopt.custom_hyperopt, HyperOptAuto)
917918
assert isinstance(hyperopt.backtesting.strategy.buy_rsi, IntParameter)

0 commit comments

Comments
 (0)