merged with feat/short
This commit is contained in:
		| @@ -98,6 +98,38 @@ class MyAwesomeStrategy(IStrategy): | ||||
| !!! Note | ||||
|     All overrides are optional and can be mixed/matched as necessary. | ||||
|  | ||||
| ### Overriding Base estimator | ||||
|  | ||||
| You can define your own estimator for Hyperopt by implementing `generate_estimator()` in the Hyperopt subclass. | ||||
|  | ||||
| ```python | ||||
| class MyAwesomeStrategy(IStrategy): | ||||
|     class HyperOpt: | ||||
|         def generate_estimator(): | ||||
|             return "RF" | ||||
|  | ||||
| ``` | ||||
|  | ||||
| Possible values are either one of "GP", "RF", "ET", "GBRT" (Details can be found in the [scikit-optimize documentation](https://scikit-optimize.github.io/)), or "an instance of a class that inherits from `RegressorMixin` (from sklearn) and where the `predict` method has an optional `return_std` argument, which returns `std(Y | x)` along with `E[Y | x]`". | ||||
|  | ||||
| Some research will be necessary to find additional Regressors. | ||||
|  | ||||
| Example for `ExtraTreesRegressor` ("ET") with additional parameters: | ||||
|  | ||||
| ```python | ||||
| class MyAwesomeStrategy(IStrategy): | ||||
|     class HyperOpt: | ||||
|         def generate_estimator(): | ||||
|             from skopt.learning import ExtraTreesRegressor | ||||
|             # Corresponds to "ET" - but allows additional parameters. | ||||
|             return ExtraTreesRegressor(n_estimators=100) | ||||
|  | ||||
| ``` | ||||
|  | ||||
| !!! Note | ||||
|     While custom estimators can be provided, it's up to you as User to do research on possible parameters and analyze / understand which ones should be used. | ||||
|     If you're unsure about this, best use one of the Defaults (`"ET"` has proven to be the most versatile) without further parameters. | ||||
|  | ||||
| ## Space options | ||||
|  | ||||
| For the additional spaces, scikit-optimize (in combination with Freqtrade) provides the following space types: | ||||
|   | ||||
| @@ -677,7 +677,7 @@ If you are optimizing ROI, Freqtrade creates the 'roi' optimization hyperspace f | ||||
|  | ||||
| These ranges should be sufficient in most cases. The minutes in the steps (ROI dict keys) are scaled linearly depending on the timeframe used. The ROI values in the steps (ROI dict values) are scaled logarithmically depending on the timeframe used. | ||||
|  | ||||
| If you have the `generate_roi_table()` and `roi_space()` methods in your custom hyperopt file, remove them in order to utilize these adaptive ROI tables and the ROI hyperoptimization space generated by Freqtrade by default. | ||||
| If you have the `generate_roi_table()` and `roi_space()` methods in your custom hyperopt, remove them in order to utilize these adaptive ROI tables and the ROI hyperoptimization space generated by Freqtrade by default. | ||||
|  | ||||
| Override the `roi_space()` method if you need components of the ROI tables to vary in other ranges. Override the `generate_roi_table()` and `roi_space()` methods and implement your own custom approach for generation of the ROI tables during hyperoptimization if you need a different structure of the ROI tables or other amount of rows (steps). | ||||
|  | ||||
|   | ||||
| @@ -165,6 +165,7 @@ Example to remove the first 10 pairs from the pairlist: | ||||
|  | ||||
| ```json | ||||
| "pairlists": [ | ||||
|     // ... | ||||
|     { | ||||
|         "method": "OffsetFilter", | ||||
|         "offset": 10 | ||||
| @@ -190,6 +191,19 @@ Sorts pairs by past trade performance, as follows: | ||||
|  | ||||
| Trade count is used as a tie breaker. | ||||
|  | ||||
| You can use the `minutes` parameter to only consider performance of the past X minutes (rolling window). | ||||
| Not defining this parameter (or setting it to 0) will use all-time performance. | ||||
|  | ||||
| ```json | ||||
| "pairlists": [ | ||||
|     // ... | ||||
|     { | ||||
|         "method": "PerformanceFilter", | ||||
|         "minutes": 1440  // rolling 24h | ||||
|     } | ||||
| ], | ||||
| ``` | ||||
|  | ||||
| !!! Note | ||||
|     `PerformanceFilter` does not support backtesting mode. | ||||
|  | ||||
|   | ||||
| @@ -15,3 +15,7 @@ For longs, the currency which pays the interest fee for the `borrowed` will alre | ||||
|     Rollover fee = P (borrowed money) * R (quat_hourly_interest) * ceiling(T/4) (in hours) | ||||
|     I (interest) = Opening fee + Rollover fee | ||||
|     [source](https://support.kraken.com/hc/en-us/articles/206161568-What-are-the-fees-for-margin-trading-) | ||||
|  | ||||
| # TODO-lev: Mention that says you can't run 2 bots on the same account with leverage, | ||||
|  | ||||
| #TODO-lev: Create a huge risk disclaimer | ||||
| @@ -288,6 +288,12 @@ Stoploss values returned from `custom_stoploss()` always specify a percentage re | ||||
|  | ||||
| The helper function [`stoploss_from_open()`](strategy-customization.md#stoploss_from_open) can be used to convert from an open price relative stop, to a current price relative stop which can be returned from `custom_stoploss()`. | ||||
|  | ||||
| ### Calculating stoploss percentage from absolute price | ||||
|  | ||||
| Stoploss values returned from `custom_stoploss()` always specify a percentage relative to `current_rate`. In order to set a stoploss at specified absolute price level, we need to use `stop_rate` to calculate what percentage relative to the `current_rate` will give you the same result as if the percentage was specified from the open price. | ||||
|  | ||||
| The helper function [`stoploss_from_absolute()`](strategy-customization.md#stoploss_from_absolute) can be used to convert from an absolute price, to a current price relative stop which can be returned from `custom_stoploss()`. | ||||
|  | ||||
| #### Stepped stoploss | ||||
|  | ||||
| Instead of continuously trailing behind the current price, this example sets fixed stoploss price levels based on the current profit. | ||||
|   | ||||
| @@ -639,6 +639,167 @@ Stoploss values returned from `custom_stoploss` must specify a percentage relati | ||||
|  | ||||
|     Full examples can be found in the [Custom stoploss](strategy-advanced.md#custom-stoploss) section of the Documentation. | ||||
|  | ||||
| !!! Note | ||||
|     Providing invalid input to `stoploss_from_open()` may produce "CustomStoploss function did not return valid stoploss" warnings. | ||||
|     This may happen if `current_profit` parameter is below specified `open_relative_stop`. Such situations may arise when closing trade | ||||
|     is blocked by `confirm_trade_exit()` method. Warnings can be solved by never blocking stop loss sells by checking `sell_reason` in | ||||
|     `confirm_trade_exit()`, or by using `return stoploss_from_open(...) or 1` idiom, which will request to not change stop loss when | ||||
|     `current_profit < open_relative_stop`. | ||||
|  | ||||
| ### *stoploss_from_absolute()* | ||||
|  | ||||
| In some situations it may be confusing to deal with stops relative to current rate. Instead, you may define a stoploss level using an absolute price. | ||||
|  | ||||
| ??? Example "Returning a stoploss using absolute price from the custom stoploss function" | ||||
|  | ||||
|     If we want to trail a stop price at 2xATR below current proce we can call `stoploss_from_absolute(current_rate - (candle['atr'] * 2), current_rate)`. | ||||
|  | ||||
|     ``` python | ||||
|  | ||||
|     from datetime import datetime | ||||
|     from freqtrade.persistence import Trade | ||||
|     from freqtrade.strategy import IStrategy, stoploss_from_open | ||||
|  | ||||
|     class AwesomeStrategy(IStrategy): | ||||
|  | ||||
|         use_custom_stoploss = True | ||||
|  | ||||
|         def populate_indicators_1h(self, dataframe: DataFrame, metadata: dict) -> DataFrame: | ||||
|             dataframe['atr'] = ta.ATR(dataframe, timeperiod=14) | ||||
|             return dataframe | ||||
|  | ||||
|         def custom_stoploss(self, pair: str, trade: 'Trade', current_time: datetime, | ||||
|                             current_rate: float, current_profit: float, **kwargs) -> float: | ||||
|             dataframe, _ = self.dp.get_analyzed_dataframe(pair, self.timeframe) | ||||
|             candle = dataframe.iloc[-1].squeeze() | ||||
|             return stoploss_from_absolute(current_rate - (candle['atr'] * 2), current_rate) | ||||
|  | ||||
|     ``` | ||||
|  | ||||
| ### *@informative()* | ||||
|  | ||||
| ``` python | ||||
| def informative(timeframe: str, asset: str = '', | ||||
|                 fmt: Optional[Union[str, Callable[[KwArg(str)], str]]] = None, | ||||
|                 ffill: bool = True) -> Callable[[PopulateIndicators], PopulateIndicators]: | ||||
|     """ | ||||
|     A decorator for populate_indicators_Nn(self, dataframe, metadata), allowing these functions to | ||||
|     define informative indicators. | ||||
|  | ||||
|     Example usage: | ||||
|  | ||||
|         @informative('1h') | ||||
|         def populate_indicators_1h(self, dataframe: DataFrame, metadata: dict) -> DataFrame: | ||||
|             dataframe['rsi'] = ta.RSI(dataframe, timeperiod=14) | ||||
|             return dataframe | ||||
|  | ||||
|     :param timeframe: Informative timeframe. Must always be equal or higher than strategy timeframe. | ||||
|     :param asset: Informative asset, for example BTC, BTC/USDT, ETH/BTC. Do not specify to use | ||||
|     current pair. | ||||
|     :param fmt: Column format (str) or column formatter (callable(name, asset, timeframe)). When not | ||||
|     specified, defaults to: | ||||
|     * {base}_{quote}_{column}_{timeframe} if asset is specified.  | ||||
|     * {column}_{timeframe} if asset is not specified. | ||||
|     Format string supports these format variables: | ||||
|     * {asset} - full name of the asset, for example 'BTC/USDT'. | ||||
|     * {base} - base currency in lower case, for example 'eth'. | ||||
|     * {BASE} - same as {base}, except in upper case. | ||||
|     * {quote} - quote currency in lower case, for example 'usdt'. | ||||
|     * {QUOTE} - same as {quote}, except in upper case. | ||||
|     * {column} - name of dataframe column. | ||||
|     * {timeframe} - timeframe of informative dataframe. | ||||
|     :param ffill: ffill dataframe after merging informative pair. | ||||
|     """ | ||||
| ``` | ||||
|  | ||||
| In most common case it is possible to easily define informative pairs by using a decorator. All decorated `populate_indicators_*` methods run in isolation, | ||||
| not having access to data from other informative pairs, in the end all informative dataframes are merged and passed to main `populate_indicators()` method. | ||||
| When hyperopting, use of hyperoptable parameter `.value` attribute is not supported. Please use `.range` attribute. See [optimizing an indicator parameter](hyperopt.md#optimizing-an-indicator-parameter) | ||||
| for more information. | ||||
|  | ||||
| ??? Example "Fast and easy way to define informative pairs" | ||||
|  | ||||
|     Most of the time we do not need power and flexibility offered by `merge_informative_pair()`, therefore we can use a decorator to quickly define informative pairs. | ||||
|  | ||||
|     ``` python | ||||
|  | ||||
|     from datetime import datetime | ||||
|     from freqtrade.persistence import Trade | ||||
|     from freqtrade.strategy import IStrategy, informative | ||||
|  | ||||
|     class AwesomeStrategy(IStrategy): | ||||
|          | ||||
|         # This method is not required.  | ||||
|         # def informative_pairs(self): ... | ||||
|  | ||||
|         # Define informative upper timeframe for each pair. Decorators can be stacked on same  | ||||
|         # method. Available in populate_indicators as 'rsi_30m' and 'rsi_1h'. | ||||
|         @informative('30m') | ||||
|         @informative('1h') | ||||
|         def populate_indicators_1h(self, dataframe: DataFrame, metadata: dict) -> DataFrame: | ||||
|             dataframe['rsi'] = ta.RSI(dataframe, timeperiod=14) | ||||
|             return dataframe | ||||
|  | ||||
|         # Define BTC/STAKE informative pair. Available in populate_indicators and other methods as | ||||
|         # 'btc_rsi_1h'. Current stake currency should be specified as {stake} format variable  | ||||
|         # instead of hardcoding actual stake currency. Available in populate_indicators and other  | ||||
|         # methods as 'btc_usdt_rsi_1h' (when stake currency is USDT). | ||||
|         @informative('1h', 'BTC/{stake}') | ||||
|         def populate_indicators_btc_1h(self, dataframe: DataFrame, metadata: dict) -> DataFrame: | ||||
|             dataframe['rsi'] = ta.RSI(dataframe, timeperiod=14) | ||||
|             return dataframe | ||||
|  | ||||
|         # Define BTC/ETH informative pair. You must specify quote currency if it is different from | ||||
|         # stake currency. Available in populate_indicators and other methods as 'eth_btc_rsi_1h'. | ||||
|         @informative('1h', 'ETH/BTC') | ||||
|         def populate_indicators_eth_btc_1h(self, dataframe: DataFrame, metadata: dict) -> DataFrame: | ||||
|             dataframe['rsi'] = ta.RSI(dataframe, timeperiod=14) | ||||
|             return dataframe | ||||
|      | ||||
|         # Define BTC/STAKE informative pair. A custom formatter may be specified for formatting | ||||
|         # column names. A callable `fmt(**kwargs) -> str` may be specified, to implement custom | ||||
|         # formatting. Available in populate_indicators and other methods as 'rsi_upper'. | ||||
|         @informative('1h', 'BTC/{stake}', '{column}') | ||||
|         def populate_indicators_btc_1h_2(self, dataframe: DataFrame, metadata: dict) -> DataFrame: | ||||
|             dataframe['rsi_upper'] = ta.RSI(dataframe, timeperiod=14) | ||||
|             return dataframe | ||||
|      | ||||
|         def populate_indicators(self, dataframe: DataFrame, metadata: dict) -> DataFrame: | ||||
|             # Strategy timeframe indicators for current pair. | ||||
|             dataframe['rsi'] = ta.RSI(dataframe, timeperiod=14) | ||||
|             # Informative pairs are available in this method. | ||||
|             dataframe['rsi_less'] = dataframe['rsi'] < dataframe['rsi_1h'] | ||||
|             return dataframe | ||||
|  | ||||
|     ``` | ||||
|  | ||||
| !!! Note | ||||
|     Do not use `@informative` decorator if you need to use data of one informative pair when generating another informative pair. Instead, define informative pairs | ||||
|     manually as described [in the DataProvider section](#complete-data-provider-sample). | ||||
|  | ||||
| !!! Note | ||||
|     Use string formatting when accessing informative dataframes of other pairs. This will allow easily changing stake currency in config without having to adjust strategy code. | ||||
|  | ||||
|     ``` python | ||||
|     def populate_buy_trend(self, dataframe: DataFrame, metadata: dict) -> DataFrame: | ||||
|         stake = self.config['stake_currency'] | ||||
|         dataframe.loc[ | ||||
|             ( | ||||
|                 (dataframe[f'btc_{stake}_rsi_1h'] < 35) | ||||
|                 & | ||||
|                 (dataframe['volume'] > 0) | ||||
|             ), | ||||
|             ['buy', 'buy_tag']] = (1, 'buy_signal_rsi') | ||||
|      | ||||
|         return dataframe | ||||
|     ``` | ||||
|  | ||||
|     Alternatively column renaming may be used to remove stake currency from column names: `@informative('1h', 'BTC/{stake}', fmt='{base}_{column}_{timeframe}')`. | ||||
|  | ||||
| !!! Warning "Duplicate method names" | ||||
|     Methods tagged with `@informative()` decorator must always have unique names! Re-using same name (for example when copy-pasting already defined informative method) | ||||
|     will overwrite previously defined method and not produce any errors due to limitations of Python programming language. In such cases you will find that indicators | ||||
|     created in earlier-defined methods are not available in the dataframe. Carefully review method names and make sure they are unique! | ||||
|  | ||||
| ## Additional data (Wallets) | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user