0fb121893d
We don't really use the salt infrastructure that we set up, which means it's cruft. Go ahead and admit that we don't use it and remove it from our systems. Change-Id: Ic43695719cbad30aded16ac480deb3dfd9b2a110
202 lines
6.1 KiB
Python
202 lines
6.1 KiB
Python
#!/usr/bin/env python
|
|
|
|
# Update the base image that is used for devstack VMs.
|
|
|
|
# Copyright (C) 2011-2012 OpenStack LLC.
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
# you may not use this file except in compliance with the License.
|
|
# You may obtain a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
|
# implied.
|
|
#
|
|
# See the License for the specific language governing permissions and
|
|
# limitations under the License.
|
|
|
|
import time
|
|
import os
|
|
import traceback
|
|
import socket
|
|
|
|
import novaclient
|
|
from novaclient.v1_1 import client as Client11
|
|
try:
|
|
from v1_0 import client as Client10
|
|
except:
|
|
pass
|
|
import paramiko
|
|
|
|
from sshclient import SSHClient
|
|
|
|
|
|
def iterate_timeout(max_seconds, purpose):
|
|
start = time.time()
|
|
count = 0
|
|
while (time.time() < start + max_seconds):
|
|
count += 1
|
|
yield count
|
|
time.sleep(2)
|
|
raise Exception("Timeout waiting for %s" % purpose)
|
|
|
|
|
|
def get_client(provider):
|
|
args = [provider.nova_username, provider.nova_api_key,
|
|
provider.nova_project_id, provider.nova_auth_url]
|
|
kwargs = {}
|
|
if provider.nova_service_type:
|
|
kwargs['service_type'] = provider.nova_service_type
|
|
if provider.nova_service_name:
|
|
kwargs['service_name'] = provider.nova_service_name
|
|
if provider.nova_service_region:
|
|
kwargs['region_name'] = provider.nova_service_region
|
|
if provider.nova_api_version == '1.0':
|
|
Client = Client10.Client
|
|
elif provider.nova_api_version == '1.1':
|
|
Client = Client11.Client
|
|
else:
|
|
raise Exception("API version not supported")
|
|
if provider.nova_rax_auth:
|
|
os.environ['NOVA_RAX_AUTH'] = '1'
|
|
client = Client(*args, **kwargs)
|
|
return client
|
|
|
|
extension_cache = {}
|
|
|
|
|
|
def get_extensions(client):
|
|
global extension_cache
|
|
cache = extension_cache.get(client)
|
|
if cache:
|
|
return cache
|
|
try:
|
|
resp, body = client.client.get('/extensions')
|
|
extensions = [x['alias'] for x in body['extensions']]
|
|
except novaclient.exceptions.NotFound:
|
|
extensions = []
|
|
extension_cache[client] = extensions
|
|
return extensions
|
|
|
|
|
|
def get_flavor(client, min_ram):
|
|
flavors = [f for f in client.flavors.list() if f.ram >= min_ram]
|
|
flavors.sort(lambda a, b: cmp(a.ram, b.ram))
|
|
return flavors[0]
|
|
|
|
|
|
def get_public_ip(server, version=4):
|
|
if 'os-floating-ips' in get_extensions(server.manager.api):
|
|
for addr in server.manager.api.floating_ips.list():
|
|
if addr.instance_id == server.id:
|
|
return addr.ip
|
|
for addr in server.addresses.get('public', []):
|
|
if type(addr) == type(u''): # Rackspace/openstack 1.0
|
|
return addr
|
|
if addr['version'] == version: # Rackspace/openstack 1.1
|
|
return addr['addr']
|
|
for addr in server.addresses.get('private', []):
|
|
# HP Cloud
|
|
if addr['version'] == version and not addr['addr'].startswith('10.'):
|
|
return addr['addr']
|
|
return None
|
|
|
|
|
|
def get_href(server):
|
|
for link in server.links:
|
|
if link['rel'] == 'self':
|
|
return link['href']
|
|
|
|
|
|
def add_public_ip(server):
|
|
ip = server.manager.api.floating_ips.create()
|
|
server.add_floating_ip(ip)
|
|
for count in iterate_timeout(600, "ip to be added"):
|
|
try:
|
|
newip = ip.manager.get(ip.id)
|
|
except:
|
|
print "Unable to get ip details, will retry"
|
|
traceback.print_exc()
|
|
time.sleep(5)
|
|
continue
|
|
|
|
if newip.instance_id == server.id:
|
|
print 'ip has been added'
|
|
return
|
|
|
|
|
|
def add_keypair(client, name):
|
|
key = paramiko.RSAKey.generate(2048)
|
|
public_key = key.get_name() + ' ' + key.get_base64()
|
|
kp = client.keypairs.create(name, public_key)
|
|
return key, kp
|
|
|
|
|
|
def wait_for_resource(wait_resource):
|
|
last_progress = None
|
|
last_status = None
|
|
# It can take a _very_ long time for Rackspace 1.0 to save an image
|
|
for count in iterate_timeout(21600, "waiting for %s" % wait_resource):
|
|
try:
|
|
resource = wait_resource.manager.get(wait_resource.id)
|
|
except:
|
|
print "Unable to list resources, will retry"
|
|
traceback.print_exc()
|
|
time.sleep(5)
|
|
continue
|
|
|
|
# In Rackspace v1.0, there is no progress attribute while queued
|
|
if hasattr(resource, 'progress'):
|
|
if (last_progress != resource.progress
|
|
or last_status != resource.status):
|
|
print resource.status, resource.progress
|
|
last_progress = resource.progress
|
|
elif last_status != resource.status:
|
|
print resource.status
|
|
last_status = resource.status
|
|
if resource.status == 'ACTIVE':
|
|
return resource
|
|
|
|
|
|
def ssh_connect(ip, username, connect_kwargs={}, timeout=60):
|
|
# HPcloud may return errno 111 for about 30 seconds after adding the IP
|
|
for count in iterate_timeout(timeout, "ssh access"):
|
|
try:
|
|
client = SSHClient(ip, username, **connect_kwargs)
|
|
break
|
|
except socket.error as e:
|
|
print "While testing ssh access:", e
|
|
time.sleep(5)
|
|
|
|
ret, out = client.ssh("echo access okay")
|
|
if "access okay" in out:
|
|
return client
|
|
return None
|
|
|
|
|
|
def delete_server(server):
|
|
try:
|
|
if 'os-floating-ips' in get_extensions(server.manager.api):
|
|
for addr in server.manager.api.floating_ips.list():
|
|
if addr.instance_id == server.id:
|
|
server.remove_floating_ip(addr)
|
|
addr.delete()
|
|
except:
|
|
print "Unable to remove floating IP"
|
|
traceback.print_exc()
|
|
|
|
try:
|
|
if 'os-keypairs' in get_extensions(server.manager.api):
|
|
for kp in server.manager.api.keypairs.list():
|
|
if kp.name == server.key_name:
|
|
kp.delete()
|
|
except:
|
|
print "Unable to delete keypair"
|
|
traceback.print_exc()
|
|
|
|
print "Deleting server", server.id
|
|
server.delete()
|