1
0
mirror of https://github.com/dkmstr/openuds.git synced 2025-01-20 14:03:49 +03:00

oVirt provider now publishes and removes poblications. Already unusable until finished oVirtLinkedDeployment at least

This commit is contained in:
Adolfo Gómez 2012-11-16 09:05:55 +00:00
parent 7717793604
commit aebacb166b
4 changed files with 181 additions and 155 deletions

View File

@ -129,6 +129,14 @@ class OVirtLinkedService(Service):
if self.baseName.value.isdigit(): if self.baseName.value.isdigit():
raise Service.ValidationException(_('The machine name can\'t be only numbers')) raise Service.ValidationException(_('The machine name can\'t be only numbers'))
def initGui(self):
'''
Loads required values inside
'''
self.ov.value = self.parent().serialize()
self.ev.value = self.parent().env().key()
machines = self.parent().getMachines() machines = self.parent().getMachines()
vals = [] vals = []
for m in machines: for m in machines:
@ -141,8 +149,43 @@ class OVirtLinkedService(Service):
vals.append( gui.choiceItem(c['id'], c['name'] ) ) vals.append( gui.choiceItem(c['id'], c['name'] ) )
self.cluster.setValues(vals) self.cluster.setValues(vals)
def sanitizeVmName(self, name):
'''
Ovirt only allows machine names with [a-zA-Z0-9_-]
'''
import re
return re.sub("[^a-zA-Z0-9_-]", "_", name)
def initGui(self): def makeTemplate(self, name, comments):
self.ov.value = self.parent().serialize() '''
self.ev.value = self.parent().env().key() Invokes makeTemplate from parent provider, completing params
Args:
name: Name to assign to template (must be previously "sanitized"
comments: Comments (UTF-8) to add to template
Returns:
template Id of the template created
Raises an exception if operation fails.
'''
return self.parent().makeTemplate(name, comments, self.machine.value, self.cluster.value, self.datastore.value)
def getTemplateState(self, templateId):
'''
Invokes getTemplateState from parent provider
Args:
templateId: templateId to remove
Returns nothing
Raises an exception if operation fails.
'''
return self.parent().getTemplateState(templateId)
def removeTemplate(self, templateId):
'''
invokes removeTemplate from parent provider
'''
return self.parent().removeTemplate(templateId)

View File

@ -232,6 +232,43 @@ class Provider(ServiceProvider):
''' '''
return self.__getApi().getStorageInfo(storageId, force) return self.__getApi().getStorageInfo(storageId, force)
def makeTemplate(self, name, comments, vmId, clusterId, storageId):
'''
Publish the machine (makes a template from it so we can create COWs) and returns the template id of
the creating machine
Args:
name: Name of the machine (care, only ascii characters and no spaces!!!)
vmId: id of the machine to be published
clusterId: id of the cluster that will hold the machine
storageId: id of the storage tuat will contain the publication AND linked clones
Returns
Raises an exception if operation could not be acomplished, or returns the id of the template being created.
'''
return self.__getApi().makeTemplate(name, comments, vmId, clusterId, storageId)
def getTemplateState(self, templateId):
'''
Returns current template state.
Returned values could be:
ok
locked
removed
(don't know if ovirt returns something more right now, will test what happens when template can't be published)
'''
return self.__getApi().getTemplateState(templateId)
def removeTemplate(self, templateId):
'''
Removes a template from ovirt server
Returns nothing, and raises an Exception if it fails
'''
return self.__getApi().removeTemplate(templateId)
@staticmethod @staticmethod
def test(env, data): def test(env, data):

View File

@ -41,43 +41,10 @@ logger = logging.getLogger(__name__)
class OVirtPublication(Publication): class OVirtPublication(Publication):
''' '''
This class shows how a publication is developed. This class provides the publication of a oVirtLinkedService
In order to a publication to work correctly, we must provide at least the
following methods:
* Of course, the __init__
* :py:meth:`.publish`
* :py:meth:`.checkState`
* :py:meth:`.finish`
Also, of course, methods from :py:class:`uds.core.Serializable.Serializable`
Publication do not have an configuration interface, all data contained
inside an instance of a Publication must be serialized if you want them between
method calls.
It's not waranteed that the class will not be serialized/deserialized
between methods calls, so, first of all, implement the marshal and umnarshal
mehods needed by all serializable classes.
Also a thing to note is that operations requested to Publications must be
*as fast as posible*. The operations executes in a separated thread,
and so it cant take a bit more time to execute, but it's recommended that
the operations executes as fast as posible, and, if it will take a long time,
split operation so we can keep track of state.
This means that, if we have "slow" operations, we must
We first of all declares an estimation of how long a publication will take.
This value is instance based, so if we override it in our class, the suggested
time could change.
The class attribute that indicates this suggested time is "suggestedTime", and
it's expressed in seconds, (i.e. "suggestedTime = 10")
''' '''
suggestedTime = 5 #: Suggested recheck time if publication is unfinished in seconds suggestedTime = 20 #: Suggested recheck time if publication is unfinished in seconds
def initialize(self): def initialize(self):
''' '''
@ -89,15 +56,17 @@ class OVirtPublication(Publication):
# We do not check anything at marshal method, so we ensure that # We do not check anything at marshal method, so we ensure that
# default values are correctly handled by marshal. # default values are correctly handled by marshal.
self._name = 'test' self._name = ''
self._reason = '' # No error, no reason for it self._reason = ''
self._number = 1 self._destroyAfter = 'f'
self._templateId = ''
self._state = 'r'
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( [self._name, self._reason, str(self._number)] ) return '\t'.join( ['v1', self._name, self._reason, self._destroyAfter, self._templateId, self._state] )
def unmarshal(self, data): def unmarshal(self, data):
''' '''
@ -105,115 +74,53 @@ class OVirtPublication(Publication):
''' '''
logger.debug('Data: {0}'.format(data)) logger.debug('Data: {0}'.format(data))
vals = data.split('\t') vals = data.split('\t')
logger.debug('Values: {0}'.format(vals)) if vals[0] == 'v1':
self._name = vals[0] self._name, self._reason, self._destroyAfter, self._templateId, self._state = vals[1:]
self._reason = vals[1]
self._number = int(vals[2])
def publish(self): def publish(self):
''' '''
This method is invoked whenever the administrator requests a new publication. Realizes the publication of the service
The method is not invoked directly (i mean, that the administration request
do no makes a call to this method), but a DelayedTask is saved witch will
initiate all publication stuff (and, of course, call this method).
You MUST implement it, so the publication do really something.
All publications can be synchronous or asynchronous.
The main difference between both is that first do whatever needed, (the
action must be fast enough to do not block core), returning State.FINISHED.
The second (asynchronous) are publications that could block the core, so
it have to be done in more than one step.
An example publication could be a copy of a virtual machine, where:
* First we invoke the copy operation to virtualization provider
* Second, we kept needed values inside instance so we can serialize
them whenever requested
* Returns an State.RUNNING, indicating the core that the publication
has started but has to finish sometime later. (We do no check
again the state and keep waiting here, because we will block the
core untill this operation is finished).
In our example wi will simple assign a name, and set number to 5. We
will use this number later, to make a "delay" at check if the publication
has finished. (see method checkState)
We also will make this publication an "stepped one", that is, it will not
finish at publish call but a later checkState call
Take care with instantiating threads from here. Whenever a publish returns
"State.RUNNING", the core will recheck it later, but not using this instance
and maybe that even do not use this server.
If you want to use threadings or somethin likt it, use DelayedTasks and
do not block it. You also musht provide the mechanism to allow those
DelayedTask to communicate with the publication.
One sample could be, for example, to copy a bunch of files, but we know
that this copy can take a long time and don't want it to take make it
all here, but in a separate task. Now, do you remember that "environment"
that is unique for every instance?, well, we can create a delayed task,
and pass that environment (owned by this intance) as a mechanism for
informing when the task is finished. (We insert at delayed tasks queue
an instance, not a class itself, so we can instantiate a class and
store it at delayed task queue.
Also note that, in that case, this class can also acomplish that by simply
using the suggestedTime attribute and the checkState method in most cases.
''' '''
self._number = 5 self._name = self.service().sanitizeVmName('UDS Publication' + ' ' + self.dsName() + "-" + str(self.revision()))
self._reason = '' comments = _('UDS pub for {0} at {1}').format( self.dsName(), str(datetime.now()).split('.')[0] )
self._reason = '' # No error, no reason for it
self._destroyAfter = 'f'
self._state = 'locked'
try:
self._templateId = self.service().makeTemplate(self._name, comments)
except Exception as e:
self._reason = str(e)
return State.ERROR
return State.RUNNING return State.RUNNING
def checkState(self): def checkState(self):
''' '''
Our publish method will initiate publication, but will not finish it. Checks state of publication creation
So in our sample, wi will only check if _number reaches 0, and if so
return that we have finished, else we will return that we are working
on it.
One publish returns State.RUNNING, this task will get called untill
checkState returns State.FINISHED.
Also, wi will make the publication fail one of every 10 calls to this
method.
Note: Destroying an publication also makes use of this method, so you
must keep the info of that you are checking (publishing or destroying...)
In our case, destroy is 1-step action so this will no get called while
destroying...
''' '''
import random if self._state == 'ok':
self._number -= 1 return State.FINISHED
# Serialization will take care of storing self._number
# One of every 10 calls if self._state == 'error':
if random.randint(0, 9) == 9:
self._reason = _('Random integer was 9!!! :-)')
return State.ERROR return State.ERROR
if self._number <= 0: self._state = self.service().getTemplateState(self._templateId)
# If publication os done (template is ready), and cancel was requested, do it just after template becomes ready
if self._state == 'ok':
if self._destroyAfter == 't':
return self.destroy()
return State.FINISHED return State.FINISHED
else:
return State.RUNNING return State.RUNNING
def finish(self): def finish(self):
''' '''
Invoked when Publication manager noticed that the publication has finished. In our case, finish does nothing
This give us the oportunity of cleaning up things (as stored vars, etc..),
or initialize variables that will be needed in a later phase (by deployed
services)
Returned value, if any, is ignored
''' '''
import string pass
import random
# Make simply a random string
self._name = ''.join(random.choice(string.ascii_uppercase + string.digits) for x in range(10))
def reasonOfError(self): def reasonOfError(self):
''' '''
@ -236,26 +143,22 @@ class OVirtPublication(Publication):
The retunred value is the same as when publishing, State.RUNNING, The retunred value is the same as when publishing, State.RUNNING,
State.FINISHED or State.ERROR. State.FINISHED or State.ERROR.
''' '''
self._name = ''
self._reason = '' # In fact, this is not needed, but cleaning up things... :-)
# We do not do anything else to destroy this instance of publication # We do not do anything else to destroy this instance of publication
return State.FINISHED if self._state == 'locked':
self._destroyAfter = 't'
return State.RUNNING
try:
self.service().removeTemplate(self._templateId)
except Exception as e:
self._reason = str(e)
return State.ERROR
return State.FINISHED
def cancel(self): def cancel(self):
''' '''
Invoked for canceling the current operation. Do same thing as destroy
This can be invoked directly by an administration or by the clean up
of the deployed service (indirectly).
When administrator requests it, the cancel is "delayed" and not
invoked directly.
Also, take into account that cancel is the initiation of, maybe, a
multiple-step action, so it returns, as publish and destroy does.
In our case, cancel simply invokes "destroy", that cleans up
things and returns that the action has finished in 1 step.
''' '''
return self.destroy() return self.destroy()

View File

@ -316,7 +316,7 @@ class Client(object):
lock.release() lock.release()
def publish(self, name, vmId, clusterId, storageId): def makeTemplate(self, name, comments, vmId, clusterId, storageId):
''' '''
Publish the machine (makes a template from it so we can create COWs) and returns the template id of Publish the machine (makes a template from it so we can create COWs) and returns the template id of
the creating machine the creating machine
@ -330,6 +330,7 @@ class Client(object):
Returns Returns
Raises an exception if operation could not be acomplished, or returns the id of the template being created. Raises an exception if operation could not be acomplished, or returns the id of the template being created.
''' '''
print "n: {0}, c: {1}, vm: {2}, cl: {3}, st: {3}".format(name, comments, vmId, clusterId, storageId)
try: try:
lock.acquire(True) lock.acquire(True)
@ -337,25 +338,67 @@ class Client(object):
api = self.__getApi() api = self.__getApi()
storage = api.storagedomains.get(id=storageId) storage = api.storagedomains.get(id=storageId)
storage_domain = params.StorageDomain(storage)
cluster = api.clusters.get(id=clusterId) cluster = api.clusters.get(id=clusterId)
vm = api.vms.get(id=vmId) vm = api.vms.get(id=vmId)
if vm.get_status().get_state() != 'down': if vm.get_status().get_state() != 'down':
raise Exception('Machine must be in down state to publish it') raise Exception('Machine must be in down state to publish it')
api.templates.add(params.Template(storage_domain=storage, origin = 'UDS', name=name, vm=vm, cluster=cluster)) template = params.Template(name=name,storage_domain=storage_domain, vm=vm, cluster=cluster, description=comments)
api.templates.add(template)
return api.templates.get(name=name).get_id() return api.templates.get(name=name).get_id()
finally: finally:
lock.release() lock.release()
def getPublishState(self, templateId): def getTemplateState(self, templateId):
'''
Returns current template state.
Returned values could be:
ok
locked
removed
(don't know if ovirt returns something more right now, will test what happens when template can't be published)
'''
try: try:
lock.acquire(True) lock.acquire(True)
api = self.__getApi() api = self.__getApi()
return api.templates.get(id=templateId).get_status().get_state() template = api.templates.get(id=templateId)
if template is None:
return 'removed'
return template.get_status().get_state()
finally: finally:
lock.release() lock.release()
def removeTemplate(self, templateId):
'''
Removes a template from ovirt server
Returns nothing, and raises an Exception if it fails
'''
try:
lock.acquire(True)
api = self.__getApi()
template = api.templates.get(id=templateId)
if template is None:
raise Exception('Template does not exists')
template.delete()
# This returns nothing, if it fails it raises an exception
finally:
lock.release()