-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcccontrol.py
241 lines (208 loc) · 8.6 KB
/
cccontrol.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
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
from optparse import OptionParser, OptionGroup
from lib.infrastructure import CCClass
from lib.cloud import CCCloud
from lib.db import CCDatabase
from lib import config
from lib import util
import logging
import sys
import os
def check_options(options):
rc = True
names = options.name
optionSet = []
optionSet.append(int(options.startClass))
optionSet.append(int(options.query))
optionSet.append(int(options.kill))
optionSet.append(int(options.addCluster))
optionSet.append(int(options.setPasswords))
optionSet.append(int(options.configureClusters))
optionSet.append(int(options.testSSHConnectivity))
if not options.database:
logger.error('you must specify a database')
rc = False
if not options.name:
logger.error('you must specify a name')
rc = False
if 1 not in optionSet:
logger.error('you must specify at least one main action')
rc = False
else:
if not reduce(lambda x, y: x^y, optionSet):
logger.error('you can only specify one main action')
rc = False
if names == '' or not names:
logger.error('you must specify a classname or classname ' + \
':clustername')
rc = False
else:
splitnames = names.split(':')
if len(splitnames) > 3:
logger.error('bad argument: %s' % names)
rc = False
if options.startClass:
if not (options.numInstances and options.numClusters):
logger.error('you must specify -c and -i when starting a class')
rc = False
else:
numInstances = int(options.numInstances)
numClusters = int(options.numClusters)
total = numInstances * numClusters
if total > 5:
print
print 'You are starting %s instances.' % total
answer = raw_input('Are you sure? (yes or no) ')
print
if (answer != 'Yes') and \
(answer != 'yes'):
rc = False
if options.kill and rc:
print
answer = raw_input('Are you sure you want to kill instances? ' + \
'(yes or no) ')
print
if (answer != 'Yes') and \
(answer != 'yes'):
rc = False
if options.setPasswords:
if options.passwordFile:
options.passwordFile = os.path.expanduser(options.passwordFile)
if not os.path.isfile(options.passwordFile):
logger.error('password file does not exist: ' + \
'%s' % options.passwordFile)
rc = False
else:
options.passwordFile = ''
if options.addCluster:
if not options.numInstances:
logger.error('you must specify -i when adding a cluster')
rc = False
else:
numInstances = int(options.numInstances)
if numInstances > 5:
print
print 'You are starting %s instances.' % numInstances
answer = raw_input('Are you sure? (yes or no) ')
print
if (answer != 'Yes') and \
(answer != 'yes'):
rc = False
return rc
def parse_options():
parser = OptionParser()
req = OptionGroup(parser, 'Required Options', 'You have to use these.')
req.add_option('-d', '--database', action='store', \
dest='database', help='location of the ' + \
'sqlite database file to use (this file will be ' + \
'created if it does not exist) [REQUIRED]')
req.add_option('-n', '--name', action='store', \
dest='name', help='specify the name as classname ' + \
'or classname:clustername [REQUIRED]')
parser.add_option_group(req)
main = OptionGroup(parser, 'Main Options', 'You must specify one ' + \
'and only one of these actions.')
main.add_option('-s', '--startClass', action='store_true', \
dest='startClass', help='launch a class of ' + \
' instances (specify the classname and -c and -i)')
main.add_option('-q', '--query', action='store_true', \
dest='query', help='query a class or cluster ')
main.add_option('-k', '--kill', action='store_true', \
dest='kill', help='kill a class or cluster ')
main.add_option('-a', '--addCluster', action='store_true', \
dest='addCluster', help='add a cluster to a class ' + \
'(specify classname and -i)')
main.add_option('-p', '--setPasswords', action='store_true', \
dest='setPasswords', help='set the root passwords ' + \
'of a class or cluster (you can specify a file ' + \
'containing passwords using -f or random passwords ' + \
'will be generated)')
main.add_option('-l', '--configureClusters', action='store_true', \
dest='configureClusters', help='configure the ' + \
'clusters /etc/hosts file and set the hostnames ' + \
'of a class or cluster')
main.add_option('-t', '--testSSHConnectivity', action='store_true', \
dest='testSSHConnectivity', help='test SSH ' + \
'connectivity to nodes')
parser.add_option_group(main)
extra = OptionGroup(parser, 'Additional Options', 'You might need these.')
extra.add_option('-c', '--numClusters', action='store', \
dest='numClusters', help='number of clusters ' + \
'to launch')
extra.add_option('-i', '--numInstances', action='store', \
dest='numInstances', help='number of instances ' + \
'per cluster to launch, c * i total instances ' + \
'will be launched, this includes the headnode')
extra.add_option('-f', '--passwordFile', action='store', \
dest='passwordFile', help='a file containing the ' + \
'passwords to set, in the format: clustername password')
extra.add_option('-v', '--verbose', action='store_true', \
dest = 'verbose', help = 'Verbose output')
extra.add_option('--no-query', action='store_false', \
dest = 'doQuery', help = 'No query before action')
parser.add_option_group(extra)
parser.set_defaults(startClass=False, \
query=False, \
kill=False, \
addCluster=False, \
setPasswords=False, \
configureClusters=False, \
testSSHConnectivity=False, \
doQuery=True)
(options, args) = parser.parse_args()
if not check_options(options):
parser.print_help()
sys.exit(1)
names = ''
if options.startClass or \
options.query or \
options.kill or \
options.addCluster or \
options.setPasswords or \
options.configureClusters or \
options.testSSHConnectivity:
names = options.name
if options.verbose:
logging.getLogger('').setLevel(logging.DEBUG)
return options, names
def main():
rc = 0
options, names = parse_options()
config_vals = config.read_config()
class_name, cluster_name, instance_name = util.get_names(names)
cccloud = CCCloud(config_vals)
db = CCDatabase(options.database)
logger.debug('db before command:')
if options.verbose:
db.print_db()
ccclass = CCClass(class_name, cluster_name, instance_name, db, \
cccloud, options.doQuery)
if options.startClass:
numClusters = int(options.numClusters)
numInstances = int(options.numInstances)
ccclass.launch(numClusters, numInstances)
elif options.query:
ccclass.query()
elif options.addCluster:
numInstances = int(options.numInstances)
ccclass.deploy_cluster(numInstances)
elif options.kill:
ccclass.kill()
elif options.setPasswords:
ccclass.set_root_passwords(options.passwordFile)
elif options.configureClusters:
ccclass.configure_hosts()
elif options.testSSHConnectivity:
ccclass.test_ssh_connectivity()
else:
logger.error('you must specify one action')
return 1
logger.debug('db after command:')
if options.verbose:
db.print_db()
db.close()
return rc
if __name__ == '__main__':
config.configure_logger()
logger = logging.getLogger('cloudcluster')
rc = main()
sys.exit(rc)