256 lines
9.5 KiB
Python
256 lines
9.5 KiB
Python
import shutil
|
|
from pathlib import Path
|
|
|
|
import ast_comments
|
|
|
|
from freqtrade.constants import Config
|
|
|
|
|
|
class StrategyUpdater:
|
|
name_mapping = {
|
|
'ticker_interval': 'timeframe',
|
|
'buy': 'enter_long',
|
|
'sell': 'exit_long',
|
|
'buy_tag': 'enter_tag',
|
|
'sell_reason': 'exit_reason',
|
|
|
|
'sell_signal': 'exit_signal',
|
|
'custom_sell': 'custom_exit',
|
|
'force_sell': 'force_exit',
|
|
'emergency_sell': 'emergency_exit',
|
|
|
|
# Strategy/config settings:
|
|
'use_sell_signal': 'use_exit_signal',
|
|
'sell_profit_only': 'exit_profit_only',
|
|
'sell_profit_offset': 'exit_profit_offset',
|
|
'ignore_roi_if_buy_signal': 'ignore_roi_if_entry_signal',
|
|
'forcebuy_enable': 'force_entry_enable',
|
|
}
|
|
|
|
function_mapping = {
|
|
'populate_buy_trend': 'populate_entry_trend',
|
|
'populate_sell_trend': 'populate_exit_trend',
|
|
'custom_sell': 'custom_exit',
|
|
'check_buy_timeout': 'check_entry_timeout',
|
|
'check_sell_timeout': 'check_exit_timeout',
|
|
# '': '',
|
|
}
|
|
# order_time_in_force, order_types, unfilledtimeout
|
|
otif_ot_unfilledtimeout = {
|
|
'buy': 'entry',
|
|
'sell': 'exit',
|
|
}
|
|
|
|
# create a dictionary that maps the old column names to the new ones
|
|
rename_dict = {'buy': 'enter_long', 'sell': 'exit_long', 'buy_tag': 'enter_tag'}
|
|
|
|
def start(self, config: Config, strategy_obj: dict) -> None:
|
|
"""
|
|
Run strategy updater
|
|
It updates a strategy to v3 with the help of the ast-module
|
|
:return: None
|
|
"""
|
|
|
|
source_file = strategy_obj['location']
|
|
strategies_backup_folder = Path.joinpath(config['user_data_dir'], "strategies_orig_updater")
|
|
target_file = Path.joinpath(strategies_backup_folder, strategy_obj['location_rel'])
|
|
|
|
# read the file
|
|
with Path(source_file).open('r') as f:
|
|
old_code = f.read()
|
|
if not strategies_backup_folder.is_dir():
|
|
Path(strategies_backup_folder).mkdir(parents=True, exist_ok=True)
|
|
|
|
# backup original
|
|
# => currently no date after the filename,
|
|
# could get overridden pretty fast if this is fired twice!
|
|
# The folder is always the same and the file name too (currently).
|
|
shutil.copy(source_file, target_file)
|
|
|
|
# update the code
|
|
new_code = self.update_code(old_code)
|
|
# write the modified code to the destination folder
|
|
with Path(source_file).open('w') as f:
|
|
f.write(new_code)
|
|
|
|
# define the function to update the code
|
|
def update_code(self, code):
|
|
# parse the code into an AST
|
|
tree = ast_comments.parse(code)
|
|
|
|
# use the AST to update the code
|
|
updated_code = self.modify_ast(tree)
|
|
|
|
# return the modified code without executing it
|
|
return updated_code
|
|
|
|
# function that uses the ast module to update the code
|
|
def modify_ast(self, tree): # noqa
|
|
# use the visitor to update the names and functions in the AST
|
|
NameUpdater().visit(tree)
|
|
|
|
# first fix the comments, so it understands "\n" properly inside multi line comments.
|
|
ast_comments.fix_missing_locations(tree)
|
|
ast_comments.increment_lineno(tree, n=1)
|
|
|
|
# generate the new code from the updated AST
|
|
# without indent {} parameters would just be written straight one after the other.
|
|
|
|
# ast_comments would be amazing since this is the only solution that carries over comments,
|
|
# but it does currently not have an unparse function, hopefully in the future ... !
|
|
# return ast_comments.unparse(tree)
|
|
|
|
return ast_comments.unparse(tree)
|
|
|
|
|
|
# Here we go through each respective node, slice, elt, key ... to replace outdated entries.
|
|
class NameUpdater(ast_comments.NodeTransformer):
|
|
def generic_visit(self, node):
|
|
|
|
# space is not yet transferred from buy/sell to entry/exit and thereby has to be skipped.
|
|
if isinstance(node, ast_comments.keyword):
|
|
if node.arg == "space":
|
|
return node
|
|
|
|
# from here on this is the original function.
|
|
for field, old_value in ast_comments.iter_fields(node):
|
|
if isinstance(old_value, list):
|
|
new_values = []
|
|
for value in old_value:
|
|
if isinstance(value, ast_comments.AST):
|
|
value = self.visit(value)
|
|
if value is None:
|
|
continue
|
|
elif not isinstance(value, ast_comments.AST):
|
|
new_values.extend(value)
|
|
continue
|
|
new_values.append(value)
|
|
old_value[:] = new_values
|
|
elif isinstance(old_value, ast_comments.AST):
|
|
new_node = self.visit(old_value)
|
|
if new_node is None:
|
|
delattr(node, field)
|
|
else:
|
|
setattr(node, field, new_node)
|
|
return node
|
|
|
|
def visit_Expr(self, node):
|
|
if hasattr(node.value, "left") and hasattr(node.value.left, "id"):
|
|
node.value.left.id = self.check_dict(StrategyUpdater.name_mapping, node.value.left.id)
|
|
self.visit(node.value)
|
|
return node
|
|
|
|
# Renames an element if contained inside a dictionary.
|
|
@staticmethod
|
|
def check_dict(current_dict: dict, element: str):
|
|
if element in current_dict:
|
|
element = current_dict[element]
|
|
return element
|
|
|
|
def visit_arguments(self, node):
|
|
if isinstance(node.args, list):
|
|
for arg in node.args:
|
|
arg.arg = self.check_dict(StrategyUpdater.name_mapping, arg.arg)
|
|
return node
|
|
|
|
def visit_Name(self, node):
|
|
# if the name is in the mapping, update it
|
|
node.id = self.check_dict(StrategyUpdater.name_mapping, node.id)
|
|
return node
|
|
|
|
def visit_Import(self, node):
|
|
# do not update the names in import statements
|
|
return node
|
|
|
|
def visit_ImportFrom(self, node):
|
|
# if hasattr(node, "module"):
|
|
# if node.module == "freqtrade.strategy.hyper":
|
|
# node.module = "freqtrade.strategy"
|
|
return node
|
|
|
|
def visit_If(self, node: ast_comments.If):
|
|
for child in ast_comments.iter_child_nodes(node):
|
|
self.visit(child)
|
|
return node
|
|
|
|
def visit_FunctionDef(self, node):
|
|
node.name = self.check_dict(StrategyUpdater.function_mapping, node.name)
|
|
self.generic_visit(node)
|
|
return node
|
|
|
|
def visit_Attribute(self, node):
|
|
if (
|
|
isinstance(node.value, ast_comments.Name)
|
|
and node.value.id == 'trades'
|
|
and node.attr == 'nr_of_successful_buys'
|
|
):
|
|
node.attr = 'nr_of_successful_entries'
|
|
return node
|
|
|
|
def visit_ClassDef(self, node):
|
|
# check if the class is derived from IStrategy
|
|
if any(isinstance(base, ast_comments.Name) and
|
|
base.id == 'IStrategy' for base in node.bases):
|
|
# check if the INTERFACE_VERSION variable exists
|
|
has_interface_version = any(
|
|
isinstance(child, ast_comments.Assign) and
|
|
isinstance(child.targets[0], ast_comments.Name) and
|
|
child.targets[0].id == 'INTERFACE_VERSION'
|
|
for child in node.body
|
|
)
|
|
|
|
# if the INTERFACE_VERSION variable does not exist, add it as the first child
|
|
if not has_interface_version:
|
|
node.body.insert(0, ast_comments.parse('INTERFACE_VERSION = 3').body[0])
|
|
# otherwise, update its value to 3
|
|
else:
|
|
for child in node.body:
|
|
if (
|
|
isinstance(child, ast_comments.Assign)
|
|
and isinstance(child.targets[0], ast_comments.Name)
|
|
and child.targets[0].id == 'INTERFACE_VERSION'
|
|
):
|
|
child.value = ast_comments.parse('3').body[0].value
|
|
self.generic_visit(node)
|
|
return node
|
|
|
|
def visit_Subscript(self, node):
|
|
if isinstance(node.slice, ast_comments.Constant):
|
|
if node.slice.value in StrategyUpdater.rename_dict:
|
|
# Replace the slice attributes with the values from rename_dict
|
|
node.slice.value = StrategyUpdater.rename_dict[node.slice.value]
|
|
if hasattr(node.slice, "elts"):
|
|
self.visit_elts(node.slice.elts)
|
|
if hasattr(node.slice, "value"):
|
|
if hasattr(node.slice.value, "elts"):
|
|
self.visit_elts(node.slice.value.elts)
|
|
return node
|
|
|
|
# elts can have elts (technically recursively)
|
|
def visit_elts(self, elts):
|
|
if isinstance(elts, list):
|
|
for elt in elts:
|
|
self.visit_elt(elt)
|
|
else:
|
|
self.visit_elt(elts)
|
|
return elts
|
|
|
|
# sub function again needed since the structure itself is highly flexible ...
|
|
def visit_elt(self, elt):
|
|
if isinstance(elt, ast_comments.Constant) and elt.value in StrategyUpdater.rename_dict:
|
|
elt.value = StrategyUpdater.rename_dict[elt.value]
|
|
if hasattr(elt, "elts"):
|
|
self.visit_elts(elt.elts)
|
|
if hasattr(elt, "args"):
|
|
if isinstance(elt.args, ast_comments.arguments):
|
|
self.visit_elts(elt.args)
|
|
else:
|
|
for arg in elt.args:
|
|
self.visit_elts(arg)
|
|
return elt
|
|
|
|
def visit_Constant(self, node):
|
|
node.value = self.check_dict(StrategyUpdater.otif_ot_unfilledtimeout, node.value)
|
|
node.value = self.check_dict(StrategyUpdater.name_mapping, node.value)
|
|
return node
|