2011-07-06 00:16:08 +00:00
|
|
|
import json
|
|
|
|
import web
|
2013-06-27 21:08:09 +00:00
|
|
|
import subprocess
|
2011-07-06 00:16:08 +00:00
|
|
|
|
|
|
|
from config import DB
|
|
|
|
|
2013-09-26 15:32:28 +00:00
|
|
|
import logging
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
2011-07-06 00:16:08 +00:00
|
|
|
def load_machine(name):
|
|
|
|
results = list(DB.select('machine', what='*',
|
|
|
|
where='name = $name',
|
|
|
|
vars=dict(name=name)))
|
|
|
|
if not results:
|
|
|
|
raise web.NotFound()
|
|
|
|
return results[0]
|
|
|
|
|
2013-06-27 21:08:09 +00:00
|
|
|
def get_sshkey(name):
|
|
|
|
if '@' in name:
|
|
|
|
_, name = name.rsplit('@')
|
|
|
|
args = ['ssh-keyscan']
|
|
|
|
args.append(name)
|
|
|
|
p = subprocess.Popen(
|
|
|
|
args=args,
|
|
|
|
stdout=subprocess.PIPE,
|
|
|
|
)
|
|
|
|
out, _ = p.communicate()
|
|
|
|
pubkey = None
|
|
|
|
for key_entry in out.splitlines():
|
|
|
|
hostname, pubkey = key_entry.split(' ', 1)
|
|
|
|
if not pubkey:
|
|
|
|
status = 1
|
|
|
|
else:
|
|
|
|
status = 0
|
|
|
|
return (pubkey), status
|
|
|
|
|
|
|
|
def update_sshkey(name, key, type):
|
|
|
|
if type == 'vps':
|
|
|
|
return
|
|
|
|
res = DB.update('machine', where='name = $name AND locked = false',
|
|
|
|
vars=dict(name=name),
|
|
|
|
sshpubkey=key,)
|
|
|
|
assert res == 1, 'Failed to update key of machine {name}'.format(name=name)
|
|
|
|
print 'Updated key on ', name
|
|
|
|
|
2011-07-06 00:16:08 +00:00
|
|
|
class MachineLock:
|
|
|
|
def GET(self, name):
|
|
|
|
row = load_machine(name)
|
|
|
|
row.locked_since = row.locked_since.isoformat()
|
|
|
|
web.header('Content-type', 'text/json')
|
|
|
|
return json.dumps(row)
|
|
|
|
|
|
|
|
def DELETE(self, name):
|
|
|
|
user = web.input('user')['user']
|
|
|
|
machine = load_machine(name)
|
|
|
|
if not machine.locked:
|
|
|
|
raise web.BadRequest()
|
|
|
|
if machine.locked_by != user:
|
|
|
|
raise web.Forbidden()
|
|
|
|
|
|
|
|
res = DB.update('machine',
|
|
|
|
where='locked = true AND name = $name AND locked_by = $user',
|
|
|
|
vars=dict(name=name, user=user),
|
2013-03-26 18:40:13 +00:00
|
|
|
locked=False, locked_by=None, description=None)
|
2011-07-06 00:16:08 +00:00
|
|
|
assert res == 1, 'Failed to unlock machine {name}'.format(name=name)
|
2013-03-25 22:01:26 +00:00
|
|
|
print user, 'unlocked', name
|
2011-07-06 00:16:08 +00:00
|
|
|
|
|
|
|
def POST(self, name):
|
|
|
|
user = web.input('user')['user']
|
2013-03-26 20:27:53 +00:00
|
|
|
desc = web.input(desc=None)['desc']
|
2011-07-06 00:16:08 +00:00
|
|
|
machine = load_machine(name)
|
|
|
|
if machine.locked:
|
|
|
|
raise web.Forbidden()
|
2013-06-27 21:08:09 +00:00
|
|
|
|
|
|
|
if machine.type == 'vps':
|
|
|
|
curkey = machine.sshpubkey
|
|
|
|
else:
|
|
|
|
curkey, getstatus = get_sshkey(name)
|
|
|
|
if getstatus != 0:
|
|
|
|
curkey = machine.sshpubkey
|
|
|
|
if machine.sshpubkey != curkey:
|
|
|
|
newkey = curkey
|
|
|
|
else:
|
|
|
|
newkey = machine.sshpubkey
|
2011-07-06 00:16:08 +00:00
|
|
|
res = DB.update('machine', where='name = $name AND locked = false',
|
|
|
|
vars=dict(name=name),
|
|
|
|
locked=True,
|
2013-03-25 23:42:59 +00:00
|
|
|
description=desc,
|
2013-06-27 21:08:09 +00:00
|
|
|
sshpubkey=newkey,
|
2011-07-06 00:16:08 +00:00
|
|
|
locked_by=user,
|
|
|
|
locked_since=web.db.SQLLiteral('NOW()'))
|
|
|
|
assert res == 1, 'Failed to lock machine {name}'.format(name=name)
|
2013-03-29 21:27:04 +00:00
|
|
|
print user, 'locked single machine', name, 'desc', desc
|
2011-07-06 00:16:08 +00:00
|
|
|
|
|
|
|
def PUT(self, name):
|
|
|
|
desc = web.input(desc=None)['desc']
|
|
|
|
status = web.input(status=None)['status']
|
2011-07-14 22:10:50 +00:00
|
|
|
sshpubkey = web.input(sshpubkey=None)['sshpubkey']
|
2011-07-06 00:16:08 +00:00
|
|
|
|
|
|
|
updated = {}
|
|
|
|
if desc is not None:
|
|
|
|
updated['description'] = desc
|
|
|
|
if status is not None:
|
|
|
|
updated['up'] = (status == 'up')
|
2011-07-14 22:10:50 +00:00
|
|
|
if sshpubkey is not None:
|
|
|
|
updated['sshpubkey'] = sshpubkey
|
2011-07-06 00:16:08 +00:00
|
|
|
|
|
|
|
if not updated:
|
|
|
|
raise web.BadRequest()
|
|
|
|
DB.update('machine', where='name = $name',
|
|
|
|
vars=dict(name=name), **updated)
|
2013-03-29 21:27:04 +00:00
|
|
|
print 'updated', name, 'with', updated, 'desc', desc
|
2011-07-06 00:16:08 +00:00
|
|
|
|
|
|
|
class Lock:
|
|
|
|
def GET(self):
|
|
|
|
rows = list(DB.select('machine', what='*'))
|
|
|
|
if not rows:
|
|
|
|
raise web.NotFound()
|
|
|
|
for row in rows:
|
|
|
|
row.locked_since = row.locked_since.isoformat()
|
|
|
|
web.header('Content-type', 'text/json')
|
|
|
|
return json.dumps(rows)
|
|
|
|
|
|
|
|
def POST(self):
|
|
|
|
user = web.input('user')['user']
|
2013-03-26 20:27:53 +00:00
|
|
|
desc = web.input(desc=None)['desc']
|
2011-07-06 00:16:08 +00:00
|
|
|
num = int(web.input('num')['num'])
|
2013-02-05 20:53:08 +00:00
|
|
|
machinetype = dict(machinetype=(web.input(machinetype='plana')['machinetype']))
|
2011-07-06 00:16:08 +00:00
|
|
|
|
|
|
|
if num < 1:
|
|
|
|
raise web.BadRequest()
|
|
|
|
|
|
|
|
tries = 0
|
2013-03-29 23:33:49 +00:00
|
|
|
check_existing = True
|
2011-07-06 00:16:08 +00:00
|
|
|
while True:
|
|
|
|
try:
|
2011-11-03 18:26:45 +00:00
|
|
|
# transaction will be rolled back if an exception is raised
|
2011-07-06 00:16:08 +00:00
|
|
|
with DB.transaction():
|
2013-03-29 23:33:49 +00:00
|
|
|
if desc is not None and check_existing:
|
|
|
|
# if a description is provided, treat it as a
|
|
|
|
# key for locking in case the same run locked
|
|
|
|
# machines in the db successfully before, but
|
|
|
|
# the web server reported failure to it
|
|
|
|
# because the request took too long. Only try
|
|
|
|
# this once per request.
|
|
|
|
check_existing = False
|
|
|
|
results = list(DB.select('machine',
|
|
|
|
machinetype, desc, user,
|
|
|
|
what='name, sshpubkey',
|
|
|
|
where='locked = true AND up = true AND type = $machinetype AND description = $desc AND locked_by = $user',
|
|
|
|
limit=num))
|
|
|
|
if len(results) == num:
|
|
|
|
name_keys = {}
|
|
|
|
for row in results:
|
|
|
|
name_keys[row.name] = row.sshpubkey
|
|
|
|
print 'reusing machines', name_keys.keys()
|
|
|
|
break
|
|
|
|
|
|
|
|
results = list(DB.select('machine', machinetype,
|
2013-06-27 21:08:09 +00:00
|
|
|
what='name, sshpubkey, type',
|
2013-03-29 23:33:49 +00:00
|
|
|
where='locked = false AND up = true AND type = $machinetype',
|
2011-07-06 00:16:08 +00:00
|
|
|
limit=num))
|
|
|
|
if len(results) < num:
|
|
|
|
raise web.HTTPError(status='503 Service Unavailable')
|
2011-07-14 22:26:49 +00:00
|
|
|
name_keys = {}
|
|
|
|
for row in results:
|
2013-06-27 21:08:09 +00:00
|
|
|
if row.type == 'vps':
|
|
|
|
curkey = row.sshpubkey
|
|
|
|
else:
|
|
|
|
curkey, getstatus = get_sshkey(row.name)
|
|
|
|
if getstatus != 0:
|
|
|
|
curkey = row.sshpubkey
|
|
|
|
if row.sshpubkey != curkey:
|
|
|
|
newkey = curkey
|
|
|
|
update_sshkey(row.name, curkey, row.type)
|
|
|
|
else:
|
|
|
|
newkey = row.sshpubkey
|
|
|
|
name_keys[row.name] = newkey
|
2011-11-03 18:26:45 +00:00
|
|
|
where_cond = web.db.sqlors('name = ', name_keys.keys()) \
|
|
|
|
+ ' AND locked = false AND up = true'
|
2011-07-14 22:26:49 +00:00
|
|
|
num_locked = DB.update('machine',
|
2011-11-03 18:26:45 +00:00
|
|
|
where=where_cond,
|
2011-07-06 00:16:08 +00:00
|
|
|
locked=True,
|
|
|
|
locked_by=user,
|
2013-03-25 23:42:59 +00:00
|
|
|
description=desc,
|
2011-07-06 00:16:08 +00:00
|
|
|
locked_since=web.db.SQLLiteral('NOW()'))
|
|
|
|
assert num_locked == num, 'Failed to lock machines'
|
2013-08-30 15:58:10 +00:00
|
|
|
except Exception:
|
|
|
|
log.exception("Saw exception")
|
2011-07-06 00:16:08 +00:00
|
|
|
tries += 1
|
|
|
|
if tries < 10:
|
|
|
|
continue
|
|
|
|
raise
|
|
|
|
else:
|
|
|
|
break
|
|
|
|
|
2013-03-29 21:27:04 +00:00
|
|
|
print user, 'locked', name_keys.keys(), 'desc', desc
|
2013-03-25 22:01:26 +00:00
|
|
|
|
2011-07-06 00:16:08 +00:00
|
|
|
web.header('Content-type', 'text/json')
|
2011-07-14 22:26:49 +00:00
|
|
|
return json.dumps(name_keys)
|