forked from TabbycatDebate/tabbycat
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdeploy_heroku.py
167 lines (144 loc) · 7.12 KB
/
deploy_heroku.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
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
#!/usr/bin/python
"""Deploys Tabbycat to Heroku.
This script is compatible with both Python 2.7 and Python 3.4 (and later)."""
import argparse
import subprocess
import re
import sys
import platform
# Arguments
parser = argparse.ArgumentParser(description="Deploy Tabbycat to a new Heroku app.")
parser.add_argument("urlname", type=str,
help="Name of the Heroku app. The app will be at urlname.herokuapp.com. Use '-' to use a Heroku-generated default.")
parser.add_argument("--no-open", action="store_false", default=True, dest="open",
help="Don't open the Heroku website in your browser at the end")
parser.add_argument("--no-init-db", action="store_false", default=True, dest="init_db",
help="Don't run initial migrations on the database")
parser.add_argument("--git-remote", type=str, default=None,
help="Name of Git remote to use. Use '-' to use the urlname. If omitted, reverts to default Heroku behaviour.")
parser.add_argument("--git-branch", type=str, default=None,
help="Git branch to push (defaults to current branch)")
parser.add_argument("--pg-plan", "--postgresql-plan", type=str, default="hobby-dev",
help="Heroku Postgres plan (default hobby-dev)")
parser.add_argument("--import-tournament", type=str, metavar="IMPORT_DIR",
help="Also run the importtournament command, importing from IMPORT_DIR")
parser.add_argument("--dry-run", action="store_true", default=False,
help="Print commands, don't run them.")
config_group = parser.add_argument_group("heroku configuration settings")
config_group.add_argument("--public-cache-timeout", type=int, default=None, metavar="TIMEOUT",
help="Set the public page cache timeout to TIMEOUT")
config_group.add_argument("--tab-cache-timeout", type=int, default=None, metavar="TIMEOUT",
help="Set the tab page cache timeout to TIMEOUT")
config_group.add_argument("--enable-debug", action="store_true", default=False,
help="Enable Django debug pages")
# Import tournament arguments are copied from importtournament.py, and should be
# updated when these options in importtournament.py change.
import_tournament_group = parser.add_argument_group("import tournament options", "Passed to the importtournament command. Ignored unless --import-tournament is used. Provided for convenience; to use other importtournament options, run the importtournament command separately instead.")
import_tournament_group.add_argument('-s', '--slug', type=str, action='store', default=None, dest="tournament_slug",
help='Override tournament slug. (Default: use name of directory.)'),
import_tournament_group.add_argument('--name', type=str, action='store', default=None, dest="tournament_name",
help='Override tournament name. (Default: use name of directory.)'),
import_tournament_group.add_argument('--short-name', type=str, action='store', default=None, dest="tournament_short_name",
help='Override tournament short name. (Default: use name of directory.)'),
args = parser.parse_args()
if platform.system() == "Windows":
subprocess_kwargs = dict(shell=True)
use_color = False
else:
subprocess_kwargs = dict()
use_color = True
# Helper functions
def print_command(command):
message = "$ " + " ".join(command)
if use_color:
message = "\033[1;36m" + message + "\033[0m"
print(message)
def run_command(command):
print_command(command)
if not args.dry_run:
subprocess.check_call(command, **subprocess_kwargs)
def run_heroku_command(command):
command.insert(0, "heroku")
command.extend(["--app", urlname])
run_command(command)
def get_output_from_command(command):
print_command(command)
output = subprocess.check_output(command, **subprocess_kwargs)
output = output.decode()
sys.stdout.write(output)
sys.stdout.flush()
return output
def print_yellow(message):
if use_color:
message = "\033[1;33m" + message + "\033[0m"
print(message)
def get_git_push_spec():
if args.git_branch:
return "master" if args.git_branch == "master" else args.git_branch + ":master"
try:
branch = get_output_from_command(["git", "symbolic-ref", "--short", "--quiet", "HEAD"]).strip()
except subprocess.CalledProcessError:
print_yellow("Attempt to find git branch name failed, trying for commit instead...")
else:
return "master" if branch == "master" else branch + ":master"
try:
return get_output_from_command(["git", "rev-parse", "--short", "--quiet", "HEAD"]).strip() + ":refs/heads/master"
except subprocess.CalledProcessError:
print_yellow("Could not determine current git commit or branch. Use --git-branch to specify a git branch to push.")
exit(1)
# Create the app with addons
addons = ["memcachier", "papertrail", "sendgrid:starter", "heroku-postgresql:%s" % args.pg_plan]
command = ["heroku", "apps:create"]
if addons:
command.extend(["--addons", ",".join(addons)])
if args.urlname != "-":
command.append(args.urlname)
output = get_output_from_command(command)
match = re.search("https://([\w_-]+)\.herokuapp\.com/\s+\|\s+(https://git.heroku.com/[\w_-]+.git)", output)
urlname = match.group(1)
heroku_url = match.group(2)
# Set config variables
command = ["config:add"]
command.append("DEBUG=1" if args.enable_debug else "DEBUG=0")
if args.public_cache_timeout:
command.append("PUBLIC_PAGE_CACHE_TIMEOUT=%d" % args.public_cache_timeout)
if args.tab_cache_timeout:
command.append("TAB_PAGES_CACHE_TIMEOUT=%d" % args.tab_cache_timeout)
command.append("NEW_RELIC_LICENSE_KEY=d73704bea90f57c35714cdf129a0680baee8eab5")
command.append("NEW_RELIC_APP_NAME=%s" % urlname)
run_heroku_command(command)
# Set up a remote, if applicable
if args.git_remote:
remote_name = args.git_remote if args.git_remote != "-" else urlname
run_heroku_command(["git:remote", "--remote", remote_name])
else:
remote_name = heroku_url
# Push source code to Heroku
push_spec = get_git_push_spec()
run_command(["git", "push", remote_name, push_spec])
if args.init_db:
# Perform initial migrations
run_heroku_command(["run", "python", "manage.py", "migrate"])
print_yellow("Now creating a superuser for the Heroku site.")
print_yellow("You'll need to respond to the prompts:")
run_heroku_command(["run", "python", "manage.py", "createsuperuser"])
# Set secret key
secret_key = get_output_from_command(["heroku", "run", "python", "manage.py", "generate_secret_key"])
secret_key = secret_key.decode().strip().split()[-1]
command = ["config:add", "DJANGO_SECRET_KEY=%s" % secret_key]
run_heroku_command(command)
# Import tournament, if provided
if args.import_tournament:
command = ["run", "python", "manage.py", "importtournament", args.import_tournament]
if args.tournament_slug:
command += ["--slug", args.tournament_slug]
if args.tournament_name:
command += ["--name", args.tournament_name]
if args.tournament_short_name:
command += ["--short-name", args.tournament_short_name]
run_heroku_command(command)
# Open in browser
if args.open:
run_heroku_command(["open"])
elif args.import_tournament:
print_yellow("Warning: You can't use --import-tournament when --no-init-db is used.")