mirror of
https://github.com/dkmstr/openuds.git
synced 2025-01-05 09:17:54 +03:00
OpenStack working (initially)
This commit is contained in:
parent
c45833c252
commit
9f4ef20dc1
@ -1,6 +1,6 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
|
||||||
VERSION=1.9.0
|
VERSION=`cat ../../VERSION`
|
||||||
RELEASE=1
|
RELEASE=1
|
||||||
|
|
||||||
top=`pwd`
|
top=`pwd`
|
||||||
|
@ -1,3 +1,15 @@
|
|||||||
|
udsactor (2.0.0) stable; urgency=medium
|
||||||
|
|
||||||
|
* Upgrade for 2.0.0
|
||||||
|
|
||||||
|
-- Adolfo Gómez García <agomez@virtualcable.es> Tue, 01 Mar 2016 03:39:21 +0100
|
||||||
|
|
||||||
|
udsactor (1.9.1) stable; urgency=medium
|
||||||
|
|
||||||
|
* Upgrade for 1.9.1
|
||||||
|
|
||||||
|
-- Adolfo Gómez García <agomez@virtualcable.es> Tue, 01 Mar 2016 03:19:21 +0100
|
||||||
|
|
||||||
udsactor (1.9.0) stable; urgency=medium
|
udsactor (1.9.0) stable; urgency=medium
|
||||||
|
|
||||||
* Upgrade for 1.9.0 (fixed package version)
|
* Upgrade for 1.9.0 (fixed package version)
|
||||||
|
@ -1,3 +1,3 @@
|
|||||||
udsactor_1.9.0_all.deb admin optional
|
udsactor-nx_2.0.0_all.deb x11 optional
|
||||||
udsactor-xrdp_1.9.0_all.deb x11 optional
|
udsactor-xrdp_2.0.0_all.deb x11 optional
|
||||||
udsactor-nx_1.9.0_all.deb x11 optional
|
udsactor_2.0.0_all.deb admin optional
|
||||||
|
@ -2,7 +2,13 @@ udsclient (2.0.0) stable; urgency=medium
|
|||||||
|
|
||||||
* Release upgrade
|
* Release upgrade
|
||||||
|
|
||||||
-- Adolfo Gómez García <agomez@virtualcable.es> Fri, 19 Feb 2015 09:33:18 +0200
|
-- Adolfo Gómez García <agomez@virtualcable.es> Tue, 01 Mar 2016 09:33:18 +0100
|
||||||
|
|
||||||
|
udsclient (1.9.1) stable; urgency=medium
|
||||||
|
|
||||||
|
* Minor fixes & making version match UDS version
|
||||||
|
|
||||||
|
-- Adolfo Gómez García <agomez@virtualcable.es> Tue, 01 Mar 2016 03:02:37 +0100
|
||||||
|
|
||||||
udsclient (1.9.0) stable; urgency=medium
|
udsclient (1.9.0) stable; urgency=medium
|
||||||
|
|
||||||
|
@ -44,7 +44,7 @@ import threading
|
|||||||
import time
|
import time
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
__updated__ = '2016-02-01'
|
__updated__ = '2016-03-07'
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -116,8 +116,7 @@ class DelayedTaskRunner(object):
|
|||||||
|
|
||||||
if taskInstance is not None:
|
if taskInstance is not None:
|
||||||
logger.debug('Executing delayedTask:>{0}<'.format(task))
|
logger.debug('Executing delayedTask:>{0}<'.format(task))
|
||||||
env = Environment.getEnvForType(taskInstance.__class__)
|
taskInstance.env = Environment.getEnvForType(taskInstance.__class__)
|
||||||
taskInstance.setEnv(env)
|
|
||||||
DelayedTaskThread(taskInstance).start()
|
DelayedTaskThread(taskInstance).start()
|
||||||
|
|
||||||
def __insert(self, instance, delay, tag):
|
def __insert(self, instance, delay, tag):
|
||||||
|
@ -33,8 +33,11 @@
|
|||||||
from __future__ import unicode_literals
|
from __future__ import unicode_literals
|
||||||
|
|
||||||
from . import action
|
from . import action
|
||||||
|
from . import common
|
||||||
|
from . import group
|
||||||
from . import process
|
from . import process
|
||||||
|
from . import publication
|
||||||
from . import userService
|
from . import userService
|
||||||
from . import servicePool
|
from . import servicePool
|
||||||
from . import task
|
from . import task
|
||||||
from . import group
|
from . import userService
|
@ -85,7 +85,7 @@ class ServiceCacheUpdater(Job):
|
|||||||
sp.userServices.update() # Cleans cached queries
|
sp.userServices.update() # Cleans cached queries
|
||||||
# If this deployedService don't have a publication active and needs it, ignore it
|
# If this deployedService don't have a publication active and needs it, ignore it
|
||||||
if sp.activePublication() is None and sp.service.getInstance().publicationType is not None:
|
if sp.activePublication() is None and sp.service.getInstance().publicationType is not None:
|
||||||
logger.debug('Needs publication but do not have one, cache test ignored')
|
logger.debug('{} Needs publication but do not have one, cache test ignored'.format(sp))
|
||||||
continue
|
continue
|
||||||
# If it has any running publication, do not generate cache anymore
|
# If it has any running publication, do not generate cache anymore
|
||||||
if sp.publications.filter(state=State.PREPARING).count() > 0:
|
if sp.publications.filter(state=State.PREPARING).count() > 0:
|
||||||
|
@ -30,13 +30,13 @@
|
|||||||
'''
|
'''
|
||||||
@author: Adolfo Gómez, dkmaster at dkmon dot com
|
@author: Adolfo Gómez, dkmaster at dkmon dot com
|
||||||
'''
|
'''
|
||||||
|
# pylint: disable=protected-access
|
||||||
from __future__ import unicode_literals
|
from __future__ import unicode_literals
|
||||||
|
|
||||||
from django.core.management.base import BaseCommand, CommandError
|
from django.core.management.base import BaseCommand # , CommandError
|
||||||
from optparse import make_option
|
from optparse import make_option
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
|
|
||||||
from django.utils.daemonize import become_daemon
|
|
||||||
from uds.core.managers.TaskManager import TaskManager
|
from uds.core.managers.TaskManager import TaskManager
|
||||||
from uds.core.util.Config import GlobalConfig
|
from uds.core.util.Config import GlobalConfig
|
||||||
import logging
|
import logging
|
||||||
@ -54,6 +54,41 @@ PID_FILE = 'taskmanager.pid'
|
|||||||
def getPidFile():
|
def getPidFile():
|
||||||
return settings.BASE_DIR + '/' + PID_FILE
|
return settings.BASE_DIR + '/' + PID_FILE
|
||||||
|
|
||||||
|
# become_daemon seems te be removed on django 1.9
|
||||||
|
# This is a copy of posix version from django 1.8
|
||||||
|
buffering = int(six.PY3)
|
||||||
|
def become_daemon(our_home_dir='.', out_log='/dev/null',
|
||||||
|
err_log='/dev/null', umask=0o022):
|
||||||
|
"Robustly turn into a UNIX daemon, running in our_home_dir."
|
||||||
|
# First fork
|
||||||
|
try:
|
||||||
|
if os.fork() > 0:
|
||||||
|
sys.exit(0) # kill off parent
|
||||||
|
except OSError as e:
|
||||||
|
sys.stderr.write("fork #1 failed: (%d) %s\n" % (e.errno, e.strerror))
|
||||||
|
sys.exit(1)
|
||||||
|
os.setsid()
|
||||||
|
os.chdir(our_home_dir)
|
||||||
|
os.umask(umask)
|
||||||
|
|
||||||
|
# Second fork
|
||||||
|
try:
|
||||||
|
if os.fork() > 0:
|
||||||
|
os._exit(0)
|
||||||
|
except OSError as e:
|
||||||
|
sys.stderr.write("fork #2 failed: (%d) %s\n" % (e.errno, e.strerror))
|
||||||
|
os._exit(1)
|
||||||
|
|
||||||
|
si = open('/dev/null', 'r')
|
||||||
|
so = open(out_log, 'a+', buffering)
|
||||||
|
se = open(err_log, 'a+', buffering)
|
||||||
|
os.dup2(si.fileno(), sys.stdin.fileno())
|
||||||
|
os.dup2(so.fileno(), sys.stdout.fileno())
|
||||||
|
os.dup2(se.fileno(), sys.stderr.fileno())
|
||||||
|
# Set custom file descriptors so that they get proper buffering.
|
||||||
|
sys.stdout, sys.stderr = so, se
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
class Command(BaseCommand):
|
class Command(BaseCommand):
|
||||||
args = "None"
|
args = "None"
|
||||||
@ -82,7 +117,7 @@ class Command(BaseCommand):
|
|||||||
|
|
||||||
pid = None
|
pid = None
|
||||||
try:
|
try:
|
||||||
pid = int(file(getPidFile(), 'r').readline())
|
pid = int(open(getPidFile(), 'r').readline())
|
||||||
except Exception:
|
except Exception:
|
||||||
pid = None
|
pid = None
|
||||||
|
|
||||||
|
@ -40,7 +40,7 @@ from uds.core.ui import gui
|
|||||||
|
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
__updated__ = '2016-03-04'
|
__updated__ = '2016-03-07'
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -161,7 +161,7 @@ class LiveService(Service):
|
|||||||
name: Name (sanitized) of the machine
|
name: Name (sanitized) of the machine
|
||||||
comments: Comments for machine
|
comments: Comments for machine
|
||||||
templateId: Id of the template to deploy from
|
templateId: Id of the template to deploy from
|
||||||
displayType: 'vnc' or 'spice'. Display to use ad oVirt admin interface
|
displayType: 'vnc' or 'spice'. Display to use ad OpenNebula admin interface
|
||||||
memoryMB: Memory requested for machine, in MB
|
memoryMB: Memory requested for machine, in MB
|
||||||
guaranteedMB: Minimum memory guaranteed for this machine
|
guaranteedMB: Minimum memory guaranteed for this machine
|
||||||
|
|
||||||
@ -198,7 +198,7 @@ class LiveService(Service):
|
|||||||
|
|
||||||
def startMachine(self, machineId):
|
def startMachine(self, machineId):
|
||||||
'''
|
'''
|
||||||
Tries to start a machine. No check is done, it is simply requested to oVirt.
|
Tries to start a machine. No check is done, it is simply requested to OpenNebula.
|
||||||
|
|
||||||
This start also "resume" suspended/paused machines
|
This start also "resume" suspended/paused machines
|
||||||
|
|
||||||
@ -211,7 +211,7 @@ class LiveService(Service):
|
|||||||
|
|
||||||
def stopMachine(self, machineId):
|
def stopMachine(self, machineId):
|
||||||
'''
|
'''
|
||||||
Tries to start a machine. No check is done, it is simply requested to oVirt
|
Tries to start a machine. No check is done, it is simply requested to OpenNebula
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
machineId: Id of the machine
|
machineId: Id of the machine
|
||||||
@ -222,7 +222,7 @@ class LiveService(Service):
|
|||||||
|
|
||||||
def suspendMachine(self, machineId):
|
def suspendMachine(self, machineId):
|
||||||
'''
|
'''
|
||||||
Tries to start a machine. No check is done, it is simply requested to oVirt
|
Tries to start a machine. No check is done, it is simply requested to OpenNebula
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
machineId: Id of the machine
|
machineId: Id of the machine
|
||||||
@ -233,7 +233,7 @@ class LiveService(Service):
|
|||||||
|
|
||||||
def removeMachine(self, machineId):
|
def removeMachine(self, machineId):
|
||||||
'''
|
'''
|
||||||
Tries to delete a machine. No check is done, it is simply requested to oVirt
|
Tries to delete a machine. No check is done, it is simply requested to OpenNebula
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
machineId: Id of the machine
|
machineId: Id of the machine
|
||||||
|
@ -39,7 +39,7 @@ from . import openStack
|
|||||||
import pickle
|
import pickle
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
__updated__ = '2016-02-26'
|
__updated__ = '2016-03-07'
|
||||||
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
@ -58,8 +58,13 @@ class LiveDeployment(UserDeployment):
|
|||||||
provider of this elements.
|
provider of this elements.
|
||||||
|
|
||||||
The logic for managing ovirt deployments (user machines in this case) is here.
|
The logic for managing ovirt deployments (user machines in this case) is here.
|
||||||
|
|
||||||
'''
|
'''
|
||||||
|
_name = ''
|
||||||
|
_ip = ''
|
||||||
|
_mac = ''
|
||||||
|
_vmid = ''
|
||||||
|
_reason = ''
|
||||||
|
_queue = None
|
||||||
|
|
||||||
# : Recheck every six seconds by default (for task methods)
|
# : Recheck every six seconds by default (for task methods)
|
||||||
suggestedTime = 6
|
suggestedTime = 6
|
||||||
@ -171,12 +176,17 @@ class LiveDeployment(UserDeployment):
|
|||||||
if self.cache.get('ready') == '1':
|
if self.cache.get('ready') == '1':
|
||||||
return State.FINISHED
|
return State.FINISHED
|
||||||
|
|
||||||
state = self.service().getMachineState(self._vmid)
|
status = self.service().getMachineState(self._vmid)
|
||||||
|
|
||||||
if state == openStack.VmState.UNKNOWN:
|
if openStack.statusIsLost(status):
|
||||||
return self.__error('Machine is not available anymore')
|
return self.__error('Machine is not available anymore')
|
||||||
|
|
||||||
self.service().startMachine()
|
if status == openStack.PAUSED:
|
||||||
|
self.service().resumeMachine(self._vmid)
|
||||||
|
elif status == openStack.STOPPED:
|
||||||
|
self.service().startMachine(self._vmId)
|
||||||
|
|
||||||
|
# Right now, we suppose the machine is ready
|
||||||
|
|
||||||
self.cache.put('ready', '1')
|
self.cache.put('ready', '1')
|
||||||
return State.FINISHED
|
return State.FINISHED
|
||||||
@ -215,25 +225,25 @@ class LiveDeployment(UserDeployment):
|
|||||||
def __initQueueForDeploy(self, forLevel2=False):
|
def __initQueueForDeploy(self, forLevel2=False):
|
||||||
|
|
||||||
if forLevel2 is False:
|
if forLevel2 is False:
|
||||||
self._queue = [opCreate, opStart, opFinish]
|
self._queue = [opCreate, opFinish]
|
||||||
else:
|
else:
|
||||||
self._queue = [opCreate, opStart, opWait, opSuspend, opFinish]
|
self._queue = [opCreate, opWait, opSuspend, opFinish]
|
||||||
|
|
||||||
def __checkMachineState(self, chkState):
|
def __checkMachineState(self, chkState):
|
||||||
logger.debug('Checking that state of machine {} ({}) is {}'.format(self._vmid, self._name, chkState))
|
logger.debug('Checking that state of machine {} ({}) is {}'.format(self._vmid, self._name, chkState))
|
||||||
state = self.service().getMachineState(self._vmid)
|
status = self.service().getMachineState(self._vmid)
|
||||||
|
|
||||||
# If we want to check an state and machine does not exists (except in case that we whant to check this)
|
# If we want to check an state and machine does not exists (except in case that we whant to check this)
|
||||||
if state == openStack.VmState.UNKNOWN:
|
if openStack.statusIsLost(status):
|
||||||
return self.__error('Machine not found')
|
return self.__error('Machine not available')
|
||||||
|
|
||||||
ret = State.RUNNING
|
ret = State.RUNNING
|
||||||
|
|
||||||
if type(chkState) is list:
|
if type(chkState) is list:
|
||||||
if state in chkState:
|
if status in chkState:
|
||||||
ret = State.FINISHED
|
ret = State.FINISHED
|
||||||
else:
|
else:
|
||||||
if state == chkState:
|
if status == chkState:
|
||||||
ret = State.FINISHED
|
ret = State.FINISHED
|
||||||
|
|
||||||
return ret
|
return ret
|
||||||
@ -340,16 +350,13 @@ class LiveDeployment(UserDeployment):
|
|||||||
if self._vmid is None:
|
if self._vmid is None:
|
||||||
raise Exception('Can\'t create machine')
|
raise Exception('Can\'t create machine')
|
||||||
|
|
||||||
# Get IP & MAC (early stage)
|
|
||||||
self._mac, self._ip = self.service().getNetInfo(self._vmid)
|
|
||||||
|
|
||||||
def __remove(self):
|
def __remove(self):
|
||||||
'''
|
'''
|
||||||
Removes a machine from system
|
Removes a machine from system
|
||||||
'''
|
'''
|
||||||
state = self.service().getMachineState(self._vmid)
|
status = self.service().getMachineState(self._vmid)
|
||||||
|
|
||||||
if state == openStack.VmState.UNKNOWN:
|
if openStack.statusIsLost(status):
|
||||||
raise Exception('Machine not found')
|
raise Exception('Machine not found')
|
||||||
|
|
||||||
self.service().removeMachine(self._vmid)
|
self.service().removeMachine(self._vmid)
|
||||||
@ -371,19 +378,25 @@ class LiveDeployment(UserDeployment):
|
|||||||
'''
|
'''
|
||||||
Checks the state of a deploy for an user or cache
|
Checks the state of a deploy for an user or cache
|
||||||
'''
|
'''
|
||||||
return self.__checkMachineState(openStack.VmState.ACTIVE)
|
ret = self.__checkMachineState(openStack.ACTIVE)
|
||||||
|
if ret == State.FINISHED:
|
||||||
|
# Get IP & MAC (early stage)
|
||||||
|
self._mac, self._ip = self.service().getNetInfo(self._vmid)
|
||||||
|
|
||||||
|
return ret
|
||||||
|
|
||||||
|
|
||||||
def __checkStart(self):
|
def __checkStart(self):
|
||||||
'''
|
'''
|
||||||
Checks if machine has started
|
Checks if machine has started
|
||||||
'''
|
'''
|
||||||
return self.__checkMachineState(openStack.VmState.ACTIVE)
|
return self.__checkMachineState(openStack.ACTIVE)
|
||||||
|
|
||||||
def __checkSuspend(self):
|
def __checkSuspend(self):
|
||||||
'''
|
'''
|
||||||
Check if the machine has suspended
|
Check if the machine has suspended
|
||||||
'''
|
'''
|
||||||
return self.__checkMachineState(openStack.VmState.SUSPENDED)
|
return self.__checkMachineState(openStack.SUSPENDED)
|
||||||
|
|
||||||
def __checkRemoved(self):
|
def __checkRemoved(self):
|
||||||
'''
|
'''
|
||||||
|
@ -35,10 +35,13 @@ from django.utils.translation import ugettext as _
|
|||||||
from uds.core.services import Publication
|
from uds.core.services import Publication
|
||||||
from uds.core.util.State import State
|
from uds.core.util.State import State
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
|
|
||||||
|
import six
|
||||||
|
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
|
|
||||||
__updated__ = '2016-02-08'
|
__updated__ = '2016-03-07'
|
||||||
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
@ -48,6 +51,11 @@ class LivePublication(Publication):
|
|||||||
'''
|
'''
|
||||||
This class provides the publication of a oVirtLinkedService
|
This class provides the publication of a oVirtLinkedService
|
||||||
'''
|
'''
|
||||||
|
_name = ''
|
||||||
|
_reason = ''
|
||||||
|
_templateId = ''
|
||||||
|
_state = 'r'
|
||||||
|
_destroyAfter = 'n'
|
||||||
|
|
||||||
suggestedTime = 2 # : Suggested recheck time if publication is unfinished in seconds
|
suggestedTime = 2 # : Suggested recheck time if publication is unfinished in seconds
|
||||||
|
|
||||||
@ -65,12 +73,13 @@ class LivePublication(Publication):
|
|||||||
self._reason = ''
|
self._reason = ''
|
||||||
self._templateId = ''
|
self._templateId = ''
|
||||||
self._state = 'r'
|
self._state = 'r'
|
||||||
|
self._destroyAfter = 'n'
|
||||||
|
|
||||||
def marshal(self):
|
def marshal(self):
|
||||||
'''
|
'''
|
||||||
returns data from an instance of Sample Publication serialized
|
returns data from an instance of Sample Publication serialized
|
||||||
'''
|
'''
|
||||||
return '\t'.join(['v1', self._name, self._reason, self._templateId, self._state])
|
return '\t'.join(['v1', self._name, self._reason, self._templateId, self._state, self._destroyAfter])
|
||||||
|
|
||||||
def unmarshal(self, data):
|
def unmarshal(self, data):
|
||||||
'''
|
'''
|
||||||
@ -79,7 +88,7 @@ class LivePublication(Publication):
|
|||||||
logger.debug('Data: {0}'.format(data))
|
logger.debug('Data: {0}'.format(data))
|
||||||
vals = data.split('\t')
|
vals = data.split('\t')
|
||||||
if vals[0] == 'v1':
|
if vals[0] == 'v1':
|
||||||
self._name, self._reason, self._templateId, self._state = vals[1:]
|
self._name, self._reason, self._templateId, self._state, self._destroyAfter = vals[1:]
|
||||||
|
|
||||||
def publish(self):
|
def publish(self):
|
||||||
'''
|
'''
|
||||||
@ -87,13 +96,16 @@ class LivePublication(Publication):
|
|||||||
'''
|
'''
|
||||||
self._name = self.service().sanitizeVmName('UDSP ' + self.dsName() + "-" + str(self.revision()))
|
self._name = self.service().sanitizeVmName('UDSP ' + self.dsName() + "-" + str(self.revision()))
|
||||||
self._reason = '' # No error, no reason for it
|
self._reason = '' # No error, no reason for it
|
||||||
self._state = 'ok'
|
self._destroyAfter = 'n'
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self._templateId = self.service().makeTemplate(self._name)
|
res = self.service().makeTemplate(self._name)
|
||||||
|
logger.debug('Result: {}'.format(res))
|
||||||
|
self._templateId = res['id']
|
||||||
|
self._state = res['status']
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self._state = 'error'
|
self._state = 'error'
|
||||||
self._reason = str(e)
|
self._reason = 'Got error {}'.format(e)
|
||||||
return State.ERROR
|
return State.ERROR
|
||||||
|
|
||||||
return State.RUNNING
|
return State.RUNNING
|
||||||
@ -105,11 +117,15 @@ class LivePublication(Publication):
|
|||||||
if self._state == 'error':
|
if self._state == 'error':
|
||||||
return State.ERROR
|
return State.ERROR
|
||||||
|
|
||||||
if self._state == 'ok':
|
if self._state == 'available':
|
||||||
return State.FINISHED
|
return State.FINISHED
|
||||||
|
|
||||||
self._state = 'ok'
|
self._state = self.service().getTemplate(self._templateId)['status'] # For next check
|
||||||
return State.FINISHED
|
|
||||||
|
if self._destroyAfter == 'y' and self._state == 'available':
|
||||||
|
return self.destroy()
|
||||||
|
|
||||||
|
return State.RUNNING
|
||||||
|
|
||||||
def finish(self):
|
def finish(self):
|
||||||
'''
|
'''
|
||||||
@ -139,11 +155,18 @@ class LivePublication(Publication):
|
|||||||
State.FINISHED or State.ERROR.
|
State.FINISHED or State.ERROR.
|
||||||
'''
|
'''
|
||||||
# We do not do anything else to destroy this instance of publication
|
# We do not do anything else to destroy this instance of publication
|
||||||
|
if self._state == 'error':
|
||||||
|
return # Nothing to cancel
|
||||||
|
|
||||||
|
if self._state == 'creating':
|
||||||
|
self._destroyAfter = 'y'
|
||||||
|
return State.RUNNING
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.service().removeTemplate(self._templateId)
|
self.service().removeTemplate(self._templateId)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self._state = 'error'
|
self._state = 'error'
|
||||||
self._reason = str(e)
|
self._reason = six.text_type(e)
|
||||||
return State.ERROR
|
return State.ERROR
|
||||||
|
|
||||||
return State.FINISHED
|
return State.FINISHED
|
||||||
|
@ -39,10 +39,10 @@ from . import helpers
|
|||||||
|
|
||||||
from uds.core.ui import gui
|
from uds.core.ui import gui
|
||||||
|
|
||||||
|
import six
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
__updated__ = '2016-03-04'
|
__updated__ = '2016-03-07'
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -105,10 +105,10 @@ class LiveService(Service):
|
|||||||
},
|
},
|
||||||
tooltip=_('Project for this service'), required=True
|
tooltip=_('Project for this service'), required=True
|
||||||
)
|
)
|
||||||
availabilityZone = gui.ChoiceField(label=_("Availability Zones"), order=3, tooltip=_('Service availability zones'), required=True)
|
availabilityZone = gui.ChoiceField(label=_("Availability Zones"), order=3, tooltip=_('Service availability zones'), required=True, rdonly=True)
|
||||||
volume = gui.ChoiceField(label=_("Volume"), order=4, tooltip=_('Base volume for service'), required=True)
|
volume = gui.ChoiceField(label=_("Volume"), order=4, tooltip=_('Base volume for service'), required=True)
|
||||||
# volumeType = gui.ChoiceField(label=_("Volume Type"), order=5, tooltip=_('Volume type for service'), required=True)
|
# volumeType = gui.ChoiceField(label=_("Volume Type"), order=5, tooltip=_('Volume type for service'), required=True)
|
||||||
networks = gui.MultiChoiceField(label=_("Networks"), order=6, tooltip=_('Networks to attach to this service'), required=True)
|
network = gui.ChoiceField(label=_("Network"), order=6, tooltip=_('Network to attach to this service'), required=True)
|
||||||
flavor = gui.ChoiceField(label=_("Flavor"), order=7, tooltip=_('Flavor for service'), required=True)
|
flavor = gui.ChoiceField(label=_("Flavor"), order=7, tooltip=_('Flavor for service'), required=True)
|
||||||
|
|
||||||
securityGroups = gui.MultiChoiceField(label=_("Security Groups"), order=8, tooltip=_('Service security groups'), required=True)
|
securityGroups = gui.MultiChoiceField(label=_("Security Groups"), order=8, tooltip=_('Service security groups'), required=True)
|
||||||
@ -137,7 +137,7 @@ class LiveService(Service):
|
|||||||
'''
|
'''
|
||||||
We check here form values to see if they are valid.
|
We check here form values to see if they are valid.
|
||||||
|
|
||||||
Note that we check them throught FROM variables, that already has been
|
Note that we check them through FROM variables, that already has been
|
||||||
initialized by __init__ method of base class, before invoking this.
|
initialized by __init__ method of base class, before invoking this.
|
||||||
'''
|
'''
|
||||||
if values is not None:
|
if values is not None:
|
||||||
@ -158,7 +158,7 @@ class LiveService(Service):
|
|||||||
Loads required values inside
|
Loads required values inside
|
||||||
'''
|
'''
|
||||||
api = self.parent().api()
|
api = self.parent().api()
|
||||||
regions = [gui.choiceItem(r, r) for r in api.listRegions()]
|
regions = [gui.choiceItem(r['id'], r['id']) for r in api.listRegions()]
|
||||||
self.region.setValues(regions)
|
self.region.setValues(regions)
|
||||||
|
|
||||||
tenants = [gui.choiceItem(t['id'], t['name']) for t in api.listProjects()]
|
tenants = [gui.choiceItem(t['id'], t['name']) for t in api.listProjects()]
|
||||||
@ -187,6 +187,12 @@ class LiveService(Service):
|
|||||||
description = 'UDS Template snapshot' if description is None else description
|
description = 'UDS Template snapshot' if description is None else description
|
||||||
return self.api.createVolumeSnapshot(self.volume.value, templateName, description)
|
return self.api.createVolumeSnapshot(self.volume.value, templateName, description)
|
||||||
|
|
||||||
|
def getTemplate(self, snapshotId):
|
||||||
|
'''
|
||||||
|
Checks current state of a template (an snapshot)
|
||||||
|
'''
|
||||||
|
return self.api.getSnapshot(snapshotId)
|
||||||
|
|
||||||
def deployFromTemplate(self, name, snapshotId):
|
def deployFromTemplate(self, name, snapshotId):
|
||||||
'''
|
'''
|
||||||
Deploys a virtual machine on selected cluster from selected template
|
Deploys a virtual machine on selected cluster from selected template
|
||||||
@ -194,45 +200,58 @@ class LiveService(Service):
|
|||||||
Args:
|
Args:
|
||||||
name: Name (sanitized) of the machine
|
name: Name (sanitized) of the machine
|
||||||
comments: Comments for machine
|
comments: Comments for machine
|
||||||
templateId: Id of the template to deploy from
|
snapshotId: Id of the snapshot to deploy from
|
||||||
displayType: 'vnc' or 'spice'. Display to use ad oVirt admin interface
|
|
||||||
memoryMB: Memory requested for machine, in MB
|
|
||||||
guaranteedMB: Minimum memory guaranteed for this machine
|
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
Id of the machine being created form template
|
Id of the machine being created form template
|
||||||
'''
|
'''
|
||||||
logger.debug('Deploying from template {0} machine {1}'.format(snapshotId, name))
|
logger.debug('Deploying from template {0} machine {1}'.format(snapshotId, name))
|
||||||
# self.datastoreHasSpace()
|
# self.datastoreHasSpace()
|
||||||
# self.api.
|
return self.api.createServerFromSnapshot(snapshotId=snapshotId,
|
||||||
|
name=name,
|
||||||
|
availabilityZone=self.availabilityZone.value,
|
||||||
|
flavorId=self.flavor.value,
|
||||||
|
networkId=self.network.value,
|
||||||
|
securityGroupsIdsList=self.securityGroups.value)['id']
|
||||||
|
|
||||||
def removeTemplate(self, templateId):
|
def removeTemplate(self, templateId):
|
||||||
'''
|
'''
|
||||||
invokes removeTemplate from parent provider
|
invokes removeTemplate from parent provider
|
||||||
'''
|
'''
|
||||||
return self.parent().removeTemplate(templateId)
|
self.api.deleteSnapshot(templateId)
|
||||||
|
|
||||||
def getMachineState(self, machineId):
|
def getMachineState(self, machineId):
|
||||||
'''
|
'''
|
||||||
Invokes getMachineState from parent provider
|
Invokes getServer from openstack client
|
||||||
(returns if machine is "active" or "inactive"
|
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
machineId: If of the machine to get state
|
machineId: If of the machine to get state
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
one of this values:
|
one of this values:
|
||||||
unassigned, down, up, powering_up, powered_down,
|
ACTIVE. The server is active.
|
||||||
paused, migrating_from, migrating_to, unknown, not_responding,
|
BUILDING. The server has not finished the original build process.
|
||||||
wait_for_launch, reboot_in_progress, saving_state, restoring_state,
|
DELETED. The server is permanently deleted.
|
||||||
suspended, image_illegal, image_locked or powering_down
|
ERROR. The server is in error.
|
||||||
Also can return'unknown' if Machine is not known
|
HARD_REBOOT. The server is hard rebooting. This is equivalent to pulling the power plug on a physical server, plugging it back in, and rebooting it.
|
||||||
|
MIGRATING. The server is being migrated to a new host.
|
||||||
|
PASSWORD. The password is being reset on the server.
|
||||||
|
PAUSED. In a paused state, the state of the server is stored in RAM. A paused server continues to run in frozen state.
|
||||||
|
REBOOT. The server is in a soft reboot state. A reboot command was passed to the operating system.
|
||||||
|
REBUILD. The server is currently being rebuilt from an image.
|
||||||
|
RESCUED. The server is in rescue mode. A rescue image is running with the original server image attached.
|
||||||
|
RESIZED. Server is performing the differential copy of data that changed during its initial copy. Server is down for this stage.
|
||||||
|
REVERT_RESIZE. The resize or migration of a server failed for some reason. The destination server is being cleaned up and the original source server is restarting.
|
||||||
|
SOFT_DELETED. The server is marked as deleted but the disk images are still available to restore.
|
||||||
|
STOPPED. The server is powered off and the disk image still persists.
|
||||||
|
SUSPENDED. The server is suspended, either by request or necessity. This status appears for only the XenServer/XCP, KVM, and ESXi hypervisors. Administrative users can suspend an instance if it is infrequently used or to perform system maintenance. When you suspend an instance, its VM state is stored on disk, all memory is written to disk, and the virtual machine is stopped. Suspending an instance is similar to placing a device in hibernation; memory and vCPUs become available to create other instances.
|
||||||
|
VERIFY_RESIZE. System is awaiting confirmation that the server is operational after a move or resize.
|
||||||
'''
|
'''
|
||||||
return self.parent().getMachineState(machineId)
|
return self.api.getServer(machineId)['status']
|
||||||
|
|
||||||
def startMachine(self, machineId):
|
def startMachine(self, machineId):
|
||||||
'''
|
'''
|
||||||
Tries to start a machine. No check is done, it is simply requested to oVirt.
|
Tries to start a machine. No check is done, it is simply requested to OpenStack.
|
||||||
|
|
||||||
This start also "resume" suspended/paused machines
|
This start also "resume" suspended/paused machines
|
||||||
|
|
||||||
@ -241,46 +260,60 @@ class LiveService(Service):
|
|||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
'''
|
'''
|
||||||
return self.parent().startMachine(machineId)
|
return self.api.startServer(machineId)
|
||||||
|
|
||||||
def stopMachine(self, machineId):
|
def stopMachine(self, machineId):
|
||||||
'''
|
'''
|
||||||
Tries to start a machine. No check is done, it is simply requested to oVirt
|
Tries to stop a machine. No check is done, it is simply requested to OpenStack
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
machineId: Id of the machine
|
machineId: Id of the machine
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
'''
|
'''
|
||||||
return self.parent().stopMachine(machineId)
|
return self.api.stopServer(machineId)
|
||||||
|
|
||||||
def suspendMachine(self, machineId):
|
def suspendMachine(self, machineId):
|
||||||
'''
|
'''
|
||||||
Tries to start a machine. No check is done, it is simply requested to oVirt
|
Tries to suspend a machine. No check is done, it is simply requested to OpenStack
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
machineId: Id of the machine
|
machineId: Id of the machine
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
'''
|
'''
|
||||||
return self.parent().suspendMachine(machineId)
|
return self.api.suspendServer(machineId)
|
||||||
|
|
||||||
|
def resumeMachine(self, machineId):
|
||||||
|
'''
|
||||||
|
Tries to start a machine. No check is done, it is simply requested to OpenStack
|
||||||
|
|
||||||
|
Args:
|
||||||
|
machineId: Id of the machine
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
'''
|
||||||
|
return self.api.suspendServer(machineId)
|
||||||
|
|
||||||
|
|
||||||
def removeMachine(self, machineId):
|
def removeMachine(self, machineId):
|
||||||
'''
|
'''
|
||||||
Tries to delete a machine. No check is done, it is simply requested to oVirt
|
Tries to delete a machine. No check is done, it is simply requested to OpenStack
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
machineId: Id of the machine
|
machineId: Id of the machine
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
'''
|
'''
|
||||||
return self.parent().removeMachine(machineId)
|
return self.api.deleteServer(machineId)
|
||||||
|
|
||||||
def getNetInfo(self, machineId, networkId=None):
|
def getNetInfo(self, machineId):
|
||||||
'''
|
'''
|
||||||
Changes the mac address of first nic of the machine to the one specified
|
Gets the mac address of first nic of the machine
|
||||||
'''
|
'''
|
||||||
return self.parent().getNetInfo(machineId, networkId=None)
|
net = self.api.getServer(machineId)['addresses']
|
||||||
|
vals = six.next(six.itervalues(net))[0] # Returns "any" mac address of any interface. We just need only one interface info
|
||||||
|
return (vals['OS-EXT-IPS-MAC:mac_addr'], vals['addr'])
|
||||||
|
|
||||||
def getBaseName(self):
|
def getBaseName(self):
|
||||||
'''
|
'''
|
||||||
@ -293,9 +326,3 @@ class LiveService(Service):
|
|||||||
Returns the length of numbers part
|
Returns the length of numbers part
|
||||||
'''
|
'''
|
||||||
return int(self.lenName.value)
|
return int(self.lenName.value)
|
||||||
|
|
||||||
def getDisplay(self):
|
|
||||||
'''
|
|
||||||
Returns the selected display type (for created machines, for administration
|
|
||||||
'''
|
|
||||||
return self.display.value
|
|
||||||
|
@ -38,7 +38,7 @@ def getResources(parameters):
|
|||||||
data = [
|
data = [
|
||||||
{'name': 'availabilityZone', 'values': zones },
|
{'name': 'availabilityZone', 'values': zones },
|
||||||
{'name': 'volume', 'values': volumes },
|
{'name': 'volume', 'values': volumes },
|
||||||
{'name': 'networks', 'values': networks },
|
{'name': 'network', 'values': networks },
|
||||||
{'name': 'flavor', 'values': flavors },
|
{'name': 'flavor', 'values': flavors },
|
||||||
{'name': 'securityGroups', 'values': securityGroups },
|
{'name': 'securityGroups', 'values': securityGroups },
|
||||||
{'name': 'volumeType', 'values': volumeTypes },
|
{'name': 'volumeType', 'values': volumeTypes },
|
||||||
|
@ -31,20 +31,17 @@
|
|||||||
.. moduleauthor:: Adolfo Gómez, dkmaster at dkmon dot com
|
.. moduleauthor:: Adolfo Gómez, dkmaster at dkmon dot com
|
||||||
'''
|
'''
|
||||||
# pylint: disable=maybe-no-member
|
# pylint: disable=maybe-no-member
|
||||||
|
from django.utils.translation import ugettext as _
|
||||||
|
|
||||||
from uds.core.util.Cache import Cache
|
from uds.core.util.Cache import Cache
|
||||||
|
|
||||||
import logging
|
import logging
|
||||||
import requests
|
import requests
|
||||||
import json
|
import json
|
||||||
import six
|
|
||||||
import hashlib
|
|
||||||
import dateutil.parser
|
import dateutil.parser
|
||||||
|
|
||||||
|
|
||||||
# Python bindings for OpenNebula
|
__updated__ = '2016-03-07'
|
||||||
from .common import sanitizeName
|
|
||||||
|
|
||||||
__updated__ = '2016-03-04'
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -64,32 +61,36 @@ VERIFY_SSL = False
|
|||||||
# Helpers
|
# Helpers
|
||||||
def ensureResponseIsValid(response, errMsg=None):
|
def ensureResponseIsValid(response, errMsg=None):
|
||||||
if response.ok is False:
|
if response.ok is False:
|
||||||
|
print "False"
|
||||||
|
try:
|
||||||
|
_x, err = response.json().popitem() # Extract any key, in case of error is expected to have only one top key so this will work
|
||||||
|
errMsg = errMsg + ': {message}'.format(**err)
|
||||||
|
except Exception:
|
||||||
|
pass # If error geting error message, simply ignore it (will be loged on service log anyway)
|
||||||
if errMsg is None:
|
if errMsg is None:
|
||||||
errMsg = 'Error checking response'
|
errMsg = 'Error checking response'
|
||||||
logger.error('{}: {}'.format(errMsg, response.content))
|
logger.error('{}: {}'.format(errMsg, response.content))
|
||||||
print response.content
|
|
||||||
print response
|
|
||||||
raise Exception(errMsg)
|
raise Exception(errMsg)
|
||||||
|
|
||||||
|
|
||||||
def getRecurringUrlJson(url, headers, key, errMsg=None):
|
def getRecurringUrlJson(url, headers, key, params=None, errMsg=None, timeout=10):
|
||||||
counter = 0
|
counter = 0
|
||||||
while True:
|
while True:
|
||||||
counter += 1
|
counter += 1
|
||||||
logger.debug('Requesting url #{}: {}'.format(counter, url))
|
logger.debug('Requesting url #{}: {} / {}'.format(counter, url, params))
|
||||||
r = requests.get(url, headers=headers, verify=VERIFY_SSL)
|
r = requests.get(url, params=params, headers=headers, verify=VERIFY_SSL, timeout=timeout)
|
||||||
|
|
||||||
ensureResponseIsValid(r, errMsg)
|
ensureResponseIsValid(r, errMsg)
|
||||||
|
|
||||||
json = r.json()
|
j = r.json()
|
||||||
|
|
||||||
for v in json[key]:
|
for v in j[key]:
|
||||||
yield v
|
yield v
|
||||||
|
|
||||||
if 'next' not in json:
|
if 'next' not in j:
|
||||||
break
|
break
|
||||||
|
|
||||||
url = json['next']
|
url = j['next']
|
||||||
|
|
||||||
|
|
||||||
# Decorators
|
# Decorators
|
||||||
@ -214,68 +215,70 @@ class Client(object):
|
|||||||
|
|
||||||
@authRequired
|
@authRequired
|
||||||
def listProjects(self):
|
def listProjects(self):
|
||||||
for p in getRecurringUrlJson(self._authUrl + 'v3/users/{user_id}/projects'.format(user_id=self._userId),
|
return getRecurringUrlJson(self._authUrl + 'v3/users/{user_id}/projects'.format(user_id=self._userId),
|
||||||
headers=self._requestHeaders(),
|
headers=self._requestHeaders(),
|
||||||
key='projects',
|
key='projects',
|
||||||
errMsg='List Projects'):
|
errMsg='List Projects',
|
||||||
|
timeout=self._timeout)
|
||||||
yield p
|
|
||||||
|
|
||||||
|
|
||||||
@authRequired
|
@authRequired
|
||||||
def listRegions(self):
|
def listRegions(self):
|
||||||
for r in getRecurringUrlJson(self._authUrl + 'v3/regions/',
|
return getRecurringUrlJson(self._authUrl + 'v3/regions/',
|
||||||
headers=self._requestHeaders(),
|
headers=self._requestHeaders(),
|
||||||
key='regions',
|
key='regions',
|
||||||
errMsg='List Regions'):
|
errMsg='List Regions',
|
||||||
yield r['id']
|
timeout=self._timeout)
|
||||||
|
|
||||||
|
|
||||||
@authProjectRequired
|
@authProjectRequired
|
||||||
def listVms(self):
|
def listServers(self, detail=False, params=None):
|
||||||
for v in getRecurringUrlJson(self._getEndpointFor('compute') + '/servers',
|
path = '/servers/' + 'detail' if detail is True else ''
|
||||||
|
return getRecurringUrlJson(self._getEndpointFor('compute') + path,
|
||||||
headers=self._requestHeaders(),
|
headers=self._requestHeaders(),
|
||||||
key='servers',
|
key='servers',
|
||||||
errMsg='List Vms'):
|
params=params,
|
||||||
yield { 'name': v['name'], 'id': v['id'] }
|
errMsg='List Vms',
|
||||||
|
timeout=self._timeout)
|
||||||
|
|
||||||
|
|
||||||
@authProjectRequired
|
@authProjectRequired
|
||||||
def listImages(self):
|
def listImages(self):
|
||||||
for i in getRecurringUrlJson(self._getEndpointFor('image') + '/v2/images?status=active',
|
return getRecurringUrlJson(self._getEndpointFor('image') + '/v2/images?status=active',
|
||||||
headers=self._requestHeaders(),
|
headers=self._requestHeaders(),
|
||||||
key='images',
|
key='images',
|
||||||
errMsg='List Images'):
|
errMsg='List Images',
|
||||||
yield { 'name': i['name'], 'size': i['size'], 'visibility': i['visibility'], 'format': i['disk_format'] }
|
timeout=self._timeout)
|
||||||
|
|
||||||
|
|
||||||
@authProjectRequired
|
@authProjectRequired
|
||||||
def listVolumeTypes(self):
|
def listVolumeTypes(self):
|
||||||
for t in getRecurringUrlJson(self._getEndpointFor('volumev2') + '/types',
|
return getRecurringUrlJson(self._getEndpointFor('volumev2') + '/types',
|
||||||
headers=self._requestHeaders(),
|
headers=self._requestHeaders(),
|
||||||
key='volume_types',
|
key='volume_types',
|
||||||
errMsg='List Volume Types'):
|
errMsg='List Volume Types',
|
||||||
yield { 'id': t['id'], 'name': t['name'] }
|
timeout=self._timeout)
|
||||||
|
|
||||||
|
|
||||||
@authProjectRequired
|
@authProjectRequired
|
||||||
def listVolumes(self):
|
def listVolumes(self):
|
||||||
# self._getEndpointFor('volumev2') + '/volumes'
|
# self._getEndpointFor('volumev2') + '/volumes'
|
||||||
for v in getRecurringUrlJson(self._getEndpointFor('volumev2') + '/volumes/detail',
|
return getRecurringUrlJson(self._getEndpointFor('volumev2') + '/volumes/detail',
|
||||||
headers=self._requestHeaders(),
|
headers=self._requestHeaders(),
|
||||||
key='volumes',
|
key='volumes',
|
||||||
errMsg='List Volumes'):
|
errMsg='List Volumes',
|
||||||
yield { 'id': v['id'], 'name': v['name'], 'size': v['size'], 'status': v['status'] }
|
timeout=self._timeout)
|
||||||
|
|
||||||
|
|
||||||
@authProjectRequired
|
@authProjectRequired
|
||||||
def listVolumeSnapshots(self, volumeId):
|
def listVolumeSnapshots(self, volumeId=None):
|
||||||
for s in getRecurringUrlJson(self._getEndpointFor('volumev2') + '/snapshots',
|
for s in getRecurringUrlJson(self._getEndpointFor('volumev2') + '/snapshots',
|
||||||
headers=self._requestHeaders(),
|
headers=self._requestHeaders(),
|
||||||
key='snapshots',
|
key='snapshots',
|
||||||
errMsg='List snapshots'):
|
errMsg='List snapshots',
|
||||||
if s['volume_id'] == volumeId:
|
timeout=self._timeout):
|
||||||
yield { 'id': s['id'], 'name': s['name'], 'description': s['description'], 'status': s['status'] }
|
if volumeId is None or s['volume_id'] == volumeId:
|
||||||
|
yield s
|
||||||
|
|
||||||
|
|
||||||
@authProjectRequired
|
@authProjectRequired
|
||||||
@ -283,43 +286,69 @@ class Client(object):
|
|||||||
for az in getRecurringUrlJson(self._getEndpointFor('compute') + '/os-availability-zone',
|
for az in getRecurringUrlJson(self._getEndpointFor('compute') + '/os-availability-zone',
|
||||||
headers=self._requestHeaders(),
|
headers=self._requestHeaders(),
|
||||||
key='availabilityZoneInfo',
|
key='availabilityZoneInfo',
|
||||||
errMsg='List Availability Zones'):
|
errMsg='List Availability Zones',
|
||||||
|
timeout=self._timeout):
|
||||||
if az['zoneState']['available'] is True:
|
if az['zoneState']['available'] is True:
|
||||||
yield az['zoneName']
|
yield az['zoneName']
|
||||||
|
|
||||||
|
|
||||||
@authProjectRequired
|
@authProjectRequired
|
||||||
def listFlavors(self):
|
def listFlavors(self):
|
||||||
for f in getRecurringUrlJson(self._getEndpointFor('compute') + '/flavors',
|
return getRecurringUrlJson(self._getEndpointFor('compute') + '/flavors',
|
||||||
headers=self._requestHeaders(),
|
headers=self._requestHeaders(),
|
||||||
key='flavors',
|
key='flavors',
|
||||||
errMsg='List Flavors'):
|
errMsg='List Flavors',
|
||||||
yield { 'id': f['id'], 'name': f['name'] }
|
timeout=self._timeout)
|
||||||
|
|
||||||
|
|
||||||
@authProjectRequired
|
@authProjectRequired
|
||||||
def listNetworks(self):
|
def listNetworks(self):
|
||||||
for n in getRecurringUrlJson(self._getEndpointFor('compute') + '/os-tenant-networks',
|
return getRecurringUrlJson(self._getEndpointFor('network') + '/v2.0/networks',
|
||||||
headers=self._requestHeaders(),
|
headers=self._requestHeaders(),
|
||||||
key='networks',
|
key='networks',
|
||||||
errMsg='List Networks'):
|
errMsg='List Networks',
|
||||||
yield { 'id': n['id'], 'name': n['label'] }
|
timeout=self._timeout)
|
||||||
|
|
||||||
|
@authProjectRequired
|
||||||
|
def listPorts(self, networkId=None, ownerId=None):
|
||||||
|
params = {}
|
||||||
|
if networkId is not None:
|
||||||
|
params['network_id'] = networkId
|
||||||
|
if ownerId is not None:
|
||||||
|
params['device_owner'] = ownerId
|
||||||
|
|
||||||
|
return getRecurringUrlJson(self._getEndpointFor('network') + '/v2.0/ports',
|
||||||
|
headers=self._requestHeaders(),
|
||||||
|
key='ports',
|
||||||
|
params=params,
|
||||||
|
errMsg='List ports',
|
||||||
|
timeout=self._timeout)
|
||||||
|
|
||||||
@authProjectRequired
|
@authProjectRequired
|
||||||
def listSecurityGroups(self):
|
def listSecurityGroups(self):
|
||||||
for s in getRecurringUrlJson(self._getEndpointFor('compute') + '/os-security-groups',
|
return getRecurringUrlJson(self._getEndpointFor('compute') + '/os-security-groups',
|
||||||
headers=self._requestHeaders(),
|
headers=self._requestHeaders(),
|
||||||
key='security_groups',
|
key='security_groups',
|
||||||
errMsg='List security groups'):
|
errMsg='List security groups',
|
||||||
yield { 'id': s['id'], 'name': s['name'] }
|
timeout=self._timeout)
|
||||||
|
|
||||||
|
|
||||||
|
@authProjectRequired
|
||||||
|
def getServer(self, serverId):
|
||||||
|
r = requests.get(self._getEndpointFor('compute') + '/servers/{server_id}'.format(server_id=serverId),
|
||||||
|
headers=self._requestHeaders(),
|
||||||
|
verify=VERIFY_SSL,
|
||||||
|
timeout=self._timeout)
|
||||||
|
|
||||||
|
ensureResponseIsValid(r, 'Get Server information')
|
||||||
|
return r.json()['server']
|
||||||
|
|
||||||
@authProjectRequired
|
@authProjectRequired
|
||||||
def getVolume(self, volumeId):
|
def getVolume(self, volumeId):
|
||||||
r = requests.get(self._getEndpointFor('volumev2') + '/volumes/{volume_id}'.format(volume_id=volumeId),
|
r = requests.get(self._getEndpointFor('volumev2') + '/volumes/{volume_id}'.format(volume_id=volumeId),
|
||||||
headers=self._requestHeaders(),
|
headers=self._requestHeaders(),
|
||||||
verify=VERIFY_SSL)
|
verify=VERIFY_SSL,
|
||||||
|
timeout=self._timeout)
|
||||||
|
|
||||||
ensureResponseIsValid(r, 'Get Volume information')
|
ensureResponseIsValid(r, 'Get Volume information')
|
||||||
|
|
||||||
@ -330,17 +359,43 @@ class Client(object):
|
|||||||
|
|
||||||
@authProjectRequired
|
@authProjectRequired
|
||||||
def getSnapshot(self, snapshotId):
|
def getSnapshot(self, snapshotId):
|
||||||
|
'''
|
||||||
|
States are:
|
||||||
|
creating, available, deleting, error, error_deleting
|
||||||
|
'''
|
||||||
r = requests.get(self._getEndpointFor('volumev2') + '/snapshots/{snapshot_id}'.format(snapshot_id=snapshotId),
|
r = requests.get(self._getEndpointFor('volumev2') + '/snapshots/{snapshot_id}'.format(snapshot_id=snapshotId),
|
||||||
headers=self._requestHeaders(),
|
headers=self._requestHeaders(),
|
||||||
verify=VERIFY_SSL)
|
verify=VERIFY_SSL,
|
||||||
|
timeout=self._timeout)
|
||||||
|
|
||||||
ensureResponseIsValid(r, 'Get Snaphost information')
|
ensureResponseIsValid(r, 'Get Snaphost information')
|
||||||
|
|
||||||
v = r.json()['snapshot']
|
v = r.json()['snapshot']
|
||||||
|
|
||||||
return { 'id': v['id'], 'name': v['name'], 'status': v['status'], 'volume_id': v['volume_id'] }
|
return v
|
||||||
|
|
||||||
|
|
||||||
|
@authProjectRequired
|
||||||
|
def updateSnapshot(self, snapshotId, name=None, description=None):
|
||||||
|
data = { 'snapshot': {} }
|
||||||
|
if name is not None:
|
||||||
|
data['snapshot']['name'] = name
|
||||||
|
|
||||||
|
if description is not None:
|
||||||
|
data['snapshot']['description'] = description
|
||||||
|
|
||||||
|
r = requests.put(self._getEndpointFor('volumev2') + '/snapshots/{snapshot_id}'.format(snapshot_id=snapshotId),
|
||||||
|
data=json.dumps(data),
|
||||||
|
headers=self._requestHeaders(),
|
||||||
|
verify=VERIFY_SSL,
|
||||||
|
timeout=self._timeout)
|
||||||
|
|
||||||
|
ensureResponseIsValid(r, 'Update Snaphost information')
|
||||||
|
|
||||||
|
v = r.json()['snapshot']
|
||||||
|
|
||||||
|
return v
|
||||||
|
|
||||||
|
|
||||||
@authProjectRequired
|
@authProjectRequired
|
||||||
def createVolumeSnapshot(self, volumeId, name, description=None):
|
def createVolumeSnapshot(self, volumeId, name, description=None):
|
||||||
@ -364,7 +419,8 @@ class Client(object):
|
|||||||
|
|
||||||
ensureResponseIsValid(r, 'Cannot create snapshot. Ensure volume is in state "available"')
|
ensureResponseIsValid(r, 'Cannot create snapshot. Ensure volume is in state "available"')
|
||||||
|
|
||||||
return r.json()
|
return r.json()['snapshot']
|
||||||
|
|
||||||
|
|
||||||
@authProjectRequired
|
@authProjectRequired
|
||||||
def createVolumeFromSnapshot(self, snapshotId, name, description=None):
|
def createVolumeFromSnapshot(self, snapshotId, name, description=None):
|
||||||
@ -373,7 +429,7 @@ class Client(object):
|
|||||||
'volume': {
|
'volume': {
|
||||||
'name': name,
|
'name': name,
|
||||||
'description': description,
|
'description': description,
|
||||||
# 'volume_type': volType, # This is the volume type, not the id
|
# 'volume_type': volType, # This seems to be the volume type name, not the id
|
||||||
'snapshot_id': snapshotId
|
'snapshot_id': snapshotId
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -388,6 +444,111 @@ class Client(object):
|
|||||||
|
|
||||||
return r.json()
|
return r.json()
|
||||||
|
|
||||||
|
@authProjectRequired
|
||||||
|
def createServerFromSnapshot(self, snapshotId, name, availabilityZone, flavorId, networkId, securityGroupsIdsList, count=1):
|
||||||
|
data = {
|
||||||
|
'server': {
|
||||||
|
'name': name,
|
||||||
|
'imageRef': '',
|
||||||
|
'os-availability-zone': availabilityZone,
|
||||||
|
'availability_zone': availabilityZone,
|
||||||
|
'block_device_mapping_v2': [{
|
||||||
|
'boot_index': '0',
|
||||||
|
'uuid': snapshotId,
|
||||||
|
# 'volume_size': 1,
|
||||||
|
# 'device_name': 'vda',
|
||||||
|
'source_type': 'snapshot',
|
||||||
|
'destination_type': 'volume',
|
||||||
|
'delete_on_termination': True
|
||||||
|
}],
|
||||||
|
'flavorRef': flavorId,
|
||||||
|
# 'OS-DCF:diskConfig': 'AUTO',
|
||||||
|
'max_count': count,
|
||||||
|
'min_count': count,
|
||||||
|
'networks': [ { 'uuid': networkId } ],
|
||||||
|
'security_groups': [{'name': sg} for sg in securityGroupsIdsList]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
r = requests.post(self._getEndpointFor('compute') + '/servers',
|
||||||
|
data=json.dumps(data),
|
||||||
|
headers=self._requestHeaders(),
|
||||||
|
verify=VERIFY_SSL,
|
||||||
|
timeout=self._timeout)
|
||||||
|
|
||||||
|
ensureResponseIsValid(r, 'Cannot create instance from snapshot.')
|
||||||
|
|
||||||
|
return r.json()['server']
|
||||||
|
|
||||||
|
|
||||||
|
@authProjectRequired
|
||||||
|
def deleteServer(self, serverId):
|
||||||
|
r = requests.post(self._getEndpointFor('compute') + '/servers/{server_id}/action'.format(server_id=serverId),
|
||||||
|
data='{"forceDelete": null}',
|
||||||
|
headers=self._requestHeaders(),
|
||||||
|
verify=VERIFY_SSL,
|
||||||
|
timeout=self._timeout)
|
||||||
|
|
||||||
|
ensureResponseIsValid(r, 'Cannot start server (probably server does not exists).')
|
||||||
|
|
||||||
|
# This does not returns anything
|
||||||
|
|
||||||
|
|
||||||
|
@authProjectRequired
|
||||||
|
def deleteSnapshot(self, snapshotId):
|
||||||
|
r = requests.delete(self._getEndpointFor('volumev2') + '/snapshots/{snapshot_id}'.format(snapshot_id=snapshotId),
|
||||||
|
headers=self._requestHeaders(),
|
||||||
|
verify=VERIFY_SSL,
|
||||||
|
timeout=self._timeout)
|
||||||
|
|
||||||
|
ensureResponseIsValid(r, 'Cannot remove snapshot.')
|
||||||
|
|
||||||
|
# Does not returns a message body
|
||||||
|
|
||||||
|
|
||||||
|
@authProjectRequired
|
||||||
|
def startServer(self, serverId):
|
||||||
|
r = requests.post(self._getEndpointFor('compute') + '/servers/{server_id}/action'.format(server_id=serverId),
|
||||||
|
data='{"os-start": null}',
|
||||||
|
headers=self._requestHeaders(),
|
||||||
|
verify=VERIFY_SSL,
|
||||||
|
timeout=self._timeout)
|
||||||
|
|
||||||
|
ensureResponseIsValid(r, 'Starting server')
|
||||||
|
|
||||||
|
# This does not returns anything
|
||||||
|
|
||||||
|
|
||||||
|
@authProjectRequired
|
||||||
|
def stopServer(self, serverId):
|
||||||
|
r = requests.post(self._getEndpointFor('compute') + '/servers/{server_id}/action'.format(server_id=serverId),
|
||||||
|
data='{"os-stop": null}',
|
||||||
|
headers=self._requestHeaders(),
|
||||||
|
verify=VERIFY_SSL,
|
||||||
|
timeout=self._timeout)
|
||||||
|
|
||||||
|
ensureResponseIsValid(r, 'Stoping server')
|
||||||
|
|
||||||
|
@authProjectRequired
|
||||||
|
def suspendServer(self, serverId):
|
||||||
|
r = requests.post(self._getEndpointFor('compute') + '/servers/{server_id}/action'.format(server_id=serverId),
|
||||||
|
data='{"suspend": null}',
|
||||||
|
headers=self._requestHeaders(),
|
||||||
|
verify=VERIFY_SSL,
|
||||||
|
timeout=self._timeout)
|
||||||
|
|
||||||
|
ensureResponseIsValid(r, 'Suspending server')
|
||||||
|
|
||||||
|
@authProjectRequired
|
||||||
|
def resumeServer(self, serverId):
|
||||||
|
r = requests.post(self._getEndpointFor('compute') + '/servers/{server_id}/action'.format(server_id=serverId),
|
||||||
|
data='{"resume": null}',
|
||||||
|
headers=self._requestHeaders(),
|
||||||
|
verify=VERIFY_SSL,
|
||||||
|
timeout=self._timeout)
|
||||||
|
|
||||||
|
ensureResponseIsValid(r, 'Resuming server')
|
||||||
|
|
||||||
|
|
||||||
def testConnection(self):
|
def testConnection(self):
|
||||||
# First, ensure requested api is supported
|
# First, ensure requested api is supported
|
||||||
@ -405,6 +566,6 @@ class Client(object):
|
|||||||
self.authPassword()
|
self.authPassword()
|
||||||
return True
|
return True
|
||||||
except Exception:
|
except Exception:
|
||||||
raise Exception('Authentication error')
|
raise Exception(_('Authentication error'))
|
||||||
|
|
||||||
raise Exception(_('Openstack does not support identity API 3.2 or newer. This OpenStack server is not compatible with UDS.'))
|
raise Exception(_('Openstack does not support identity API 3.2 or newer. This OpenStack server is not compatible with UDS.'))
|
||||||
|
@ -41,7 +41,7 @@ import six
|
|||||||
from defusedxml import minidom
|
from defusedxml import minidom
|
||||||
|
|
||||||
|
|
||||||
__updated__ = '2016-03-04'
|
__updated__ = '2016-03-07'
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -50,7 +50,3 @@ from .common import *
|
|||||||
|
|
||||||
from .UDSOpenStackClient import Client
|
from .UDSOpenStackClient import Client
|
||||||
|
|
||||||
from . import template
|
|
||||||
from . import vm
|
|
||||||
from . import storage
|
|
||||||
|
|
||||||
|
@ -33,15 +33,26 @@
|
|||||||
|
|
||||||
import re
|
import re
|
||||||
|
|
||||||
from libcloud.compute.types import Provider
|
|
||||||
from libcloud.compute.providers import get_driver
|
|
||||||
|
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
__updated__ = '2016-03-04'
|
__updated__ = '2016-03-07'
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
(ACTIVE, BUILDING, DELETED, ERROR,
|
||||||
|
HARD_REBOOT, MIGRATING, PASSWORD,
|
||||||
|
PAUSED, REBOOT, REBUILD, RESCUED,
|
||||||
|
RESIZED, REVERT_RESIZE, SOFT_DELETED,
|
||||||
|
STOPPED, SUSPENDED, UNKNOWN, VERIFY_RESIZE) = ('ACTIVE', 'BUILDING', 'DELETED', 'ERROR',
|
||||||
|
'HARD_REBOOT', 'MIGRATING', 'PASSWORD',
|
||||||
|
'PAUSED', 'REBOOT', 'REBUILD', 'RESCUED',
|
||||||
|
'RESIZED', 'REVERT_RESIZE', 'SOFT_DELETED',
|
||||||
|
'STOPPED', 'SUSPENDED', 'UNKNOWN', 'VERIFY_RESIZE')
|
||||||
|
|
||||||
|
# Helpers to check statuses
|
||||||
|
def statusIsLost(status):
|
||||||
|
return status in [DELETED, ERROR, UNKNOWN, SOFT_DELETED]
|
||||||
|
|
||||||
def sanitizeName(name):
|
def sanitizeName(name):
|
||||||
'''
|
'''
|
||||||
machine names with [a-zA-Z0-9_-]
|
machine names with [a-zA-Z0-9_-]
|
||||||
|
@ -1,42 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
|
|
||||||
#
|
|
||||||
# Copyright (c) 2012 Virtual Cable S.L.
|
|
||||||
# All rights reserved.
|
|
||||||
#
|
|
||||||
# Redistribution and use in source and binary forms, with or without modification,
|
|
||||||
# are permitted provided that the following conditions are met:
|
|
||||||
#
|
|
||||||
# * Redistributions of source code must retain the above copyright notice,
|
|
||||||
# this list of conditions and the following disclaimer.
|
|
||||||
# * Redistributions in binary form must reproduce the above copyright notice,
|
|
||||||
# this list of conditions and the following disclaimer in the documentation
|
|
||||||
# and/or other materials provided with the distribution.
|
|
||||||
# * Neither the name of Virtual Cable S.L. nor the names of its contributors
|
|
||||||
# may be used to endorse or promote products derived from this software
|
|
||||||
# without specific prior written permission.
|
|
||||||
#
|
|
||||||
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
||||||
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
||||||
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
|
||||||
# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
||||||
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
||||||
# SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
|
||||||
# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
||||||
# OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
||||||
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
|
|
||||||
'''
|
|
||||||
.. moduleauthor:: Adolfo Gómez, dkmaster at dkmon dot com
|
|
||||||
'''
|
|
||||||
|
|
||||||
import logging
|
|
||||||
import six
|
|
||||||
import oca
|
|
||||||
|
|
||||||
|
|
||||||
__updated__ = '2016-02-25'
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
|
||||||
|
|
@ -1,46 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
|
|
||||||
#
|
|
||||||
# Copyright (c) 2012 Virtual Cable S.L.
|
|
||||||
# All rights reserved.
|
|
||||||
#
|
|
||||||
# Redistribution and use in source and binary forms, with or without modification,
|
|
||||||
# are permitted provided that the following conditions are met:
|
|
||||||
#
|
|
||||||
# * Redistributions of source code must retain the above copyright notice,
|
|
||||||
# this list of conditions and the following disclaimer.
|
|
||||||
# * Redistributions in binary form must reproduce the above copyright notice,
|
|
||||||
# this list of conditions and the following disclaimer in the documentation
|
|
||||||
# and/or other materials provided with the distribution.
|
|
||||||
# * Neither the name of Virtual Cable S.L. nor the names of its contributors
|
|
||||||
# may be used to endorse or promote products derived from this software
|
|
||||||
# without specific prior written permission.
|
|
||||||
#
|
|
||||||
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
||||||
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
||||||
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
|
||||||
# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
||||||
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
||||||
# SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
|
||||||
# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
||||||
# OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
||||||
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
|
|
||||||
'''
|
|
||||||
.. moduleauthor:: Adolfo Gómez, dkmaster at dkmon dot com
|
|
||||||
'''
|
|
||||||
|
|
||||||
import logging
|
|
||||||
import six
|
|
||||||
import oca
|
|
||||||
|
|
||||||
from defusedxml import minidom
|
|
||||||
# Python bindings for OpenNebula
|
|
||||||
from .common import sanitizeName
|
|
||||||
|
|
||||||
__updated__ = '2016-02-25'
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
@ -1,45 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
|
|
||||||
#
|
|
||||||
# Copyright (c) 2012 Virtual Cable S.L.
|
|
||||||
# All rights reserved.
|
|
||||||
#
|
|
||||||
# Redistribution and use in source and binary forms, with or without modification,
|
|
||||||
# are permitted provided that the following conditions are met:
|
|
||||||
#
|
|
||||||
# * Redistributions of source code must retain the above copyright notice,
|
|
||||||
# this list of conditions and the following disclaimer.
|
|
||||||
# * Redistributions in binary form must reproduce the above copyright notice,
|
|
||||||
# this list of conditions and the following disclaimer in the documentation
|
|
||||||
# and/or other materials provided with the distribution.
|
|
||||||
# * Neither the name of Virtual Cable S.L. nor the names of its contributors
|
|
||||||
# may be used to endorse or promote products derived from this software
|
|
||||||
# without specific prior written permission.
|
|
||||||
#
|
|
||||||
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
||||||
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
||||||
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
|
||||||
# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
||||||
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
||||||
# SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
|
||||||
# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
||||||
# OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
||||||
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
|
|
||||||
'''
|
|
||||||
.. moduleauthor:: Adolfo Gómez, dkmaster at dkmon dot com
|
|
||||||
'''
|
|
||||||
|
|
||||||
import logging
|
|
||||||
import six
|
|
||||||
import oca
|
|
||||||
|
|
||||||
from defusedxml import minidom
|
|
||||||
# Python bindings for OpenNebula
|
|
||||||
|
|
||||||
__updated__ = '2016-02-25'
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue
Block a user