-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathmain.py
122 lines (108 loc) · 4.86 KB
/
main.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
"""
The main program.
1. Log in to fantasy.premierleague.com.
2. Get the current squad.
3. Calculate the best squad possible for next week.
4. Update the squad on fantasy.premierleague.com.
5. Calculate the best possible starting lineup for next week.
6. Update the starting lineup on fantasy.premierleague.com.
"""
import argparse
import codecs
import constants
import linear_solver
import logging
import neural_network
import sys
import web_service
import getpass
import datetime
logLevels = {
'debug': logging.DEBUG,
'info': logging.INFO,
'warn': logging.WARN
}
# Set up the command line parser
parser = argparse.ArgumentParser(description='Mr Robot v3.0')
parser.add_argument('username', help='Login username for fantasy.premierleague.com')
parser.add_argument('--password', help='Login password for fantasy.premierleague.com')
parser.add_argument('--apply', action='store_true', help='Whether to apply the changes (default: False)')
parser.add_argument('--check-deadline', action='store_true', help='Check the deadline is the same day before running the script (default: False)')
parser.add_argument('--wildcard', action='store_true', help='Use to ignore transfer costs when calculating the new lineup (default: False)')
parser.add_argument('--ignore-squad', action='store_true', help='Whether to ignore the current squad when calculating the new squad (default: False)')
parser.add_argument('--update-model', action='store_true', help='Whether to recalculate the model or use the stored one. Note: this can take a long time! (default: False)')
parser.add_argument('--log-level', choices=list(logLevels.keys()), help='Set the logging level (default: "info")', default='info')
args = parser.parse_args()
if not args.password:
args.password = getpass.getpass(prompt='Password for {}: '.format(constants.LOGIN_URL))
# Set up the logger
# Log info to stdout, debug to file
fileHandler = logging.FileHandler('./.debug.log', 'w')
fileHandler.setLevel(logging.DEBUG)
consoleHandler = logging.StreamHandler(sys.stdout)
consoleHandler.setLevel(logLevels[args.log_level])
logging.basicConfig(
handlers=[
consoleHandler,
fileHandler
],
level=logging.DEBUG,
format="[%(asctime)s] %(levelname)s [%(name)s.%(funcName)s:%(lineno)d] %(message)s",
)
logger = logging.getLogger()
# Use a StreamWriter to output in UTF-8 else some of the logs can cause errors
# This may result in some characters not rendering correctly in Windows cmd window
# http://stackoverflow.com/questions/16346914/python-3-2-unicodeencodeerror-charmap-codec-cant-encode-character-u2013-i
if sys.stdout.encoding != 'UTF-8':
sys.stdout = codecs.getwriter('utf-8')(sys.stdout.buffer, 'strict')
if sys.stderr.encoding != 'UTF-8':
sys.stderr = codecs.getwriter('utf-8')(sys.stderr.buffer, 'strict')
# Login
logger.info('Logging in to {}'.format(constants.LOGIN_URL))
web_service.login(args.username, args.password)
if args.check_deadline:
# Check deadline is today
logger.info('Checking the deadline')
today = datetime.datetime.now().strftime('%Y-%m-%d')
logger.debug('Today is {}'.format(today))
deadline = web_service.get_deadline_date()
if not deadline == today:
logger.info('Deadline is not today, exiting')
sys.exit()
else:
logger.info('Deadline is today, continuing')
# Initialise the neural network
logger.info('Initialising the neural network')
if args.update_model:
logger.info('Updating the model')
neural_network.train_model()
neural_network.test_model()
neural_network.save_model()
else:
logger.info('Loading model')
neural_network.load_model()
# Get the current squad
if not args.ignore_squad:
logger.info('Retrieving the current squad')
CURRENT_SQUAD = web_service.get_transfers_squad()
# Calculate the new squad
logger.info('Calculating the new squad')
if args.ignore_squad:
NEW_SQUAD = linear_solver.select_squad_ignore_transfers(constants.INITIAL_TEAM_VALUE)
else:
NEW_SQUAD = linear_solver.select_squad(CURRENT_SQUAD, args.wildcard)
# Calculate the new starting lineup
logger.info('Calculating the new starting lineup')
NEW_STARTING = linear_solver.select_starting(NEW_SQUAD)
if args.apply or input('Apply these changes? (y/n): ').lower().strip() == 'y':
# make transfers to update the squad on fantasy.premierleague.com
logger.info('Applying the transfers')
WILDCARD_STATUS = (next(x for x in CURRENT_SQUAD['chips'] if x['name'] == 'wildcard'))['status_for_entry'] == 'available'
TRANSFER_OBJECT = web_service.create_transfers_object(
CURRENT_SQUAD['picks'], NEW_SQUAD, args.wildcard or ((constants.NUM_CHANGES >= 6) and WILDCARD_STATUS))
web_service.make_transfers(TRANSFER_OBJECT)
# update the starting lineup on fantasy.premierleague.com
logger.info('Updating the starting lineup')
web_service.set_starting_lineup(NEW_STARTING)
else:
logger.info('Changes not applied')