Refactoring, to make things easier to understand and use

This commit is contained in:
Adolfo Gómez 2012-07-20 01:38:15 +00:00
parent bef6a52354
commit 713614a784
18 changed files with 100 additions and 161 deletions

View File

@ -53,6 +53,7 @@ encoding//src/uds/core/managers/UserPrefsManager.py=utf-8
encoding//src/uds/core/managers/UserServiceManager.py=utf-8 encoding//src/uds/core/managers/UserServiceManager.py=utf-8
encoding//src/uds/core/osmanagers/BaseOsManager.py=utf-8 encoding//src/uds/core/osmanagers/BaseOsManager.py=utf-8
encoding//src/uds/core/osmanagers/OSManagersFactory.py=utf-8 encoding//src/uds/core/osmanagers/OSManagersFactory.py=utf-8
encoding//src/uds/core/osmanagers/__init__.py=utf-8
encoding//src/uds/core/services/BaseDeployed.py=utf-8 encoding//src/uds/core/services/BaseDeployed.py=utf-8
encoding//src/uds/core/services/BasePublication.py=utf-8 encoding//src/uds/core/services/BasePublication.py=utf-8
encoding//src/uds/core/services/BaseService.py=utf-8 encoding//src/uds/core/services/BaseService.py=utf-8
@ -96,8 +97,6 @@ encoding//src/uds/migrations/0005_auto__add_field_config_crypt.py=utf-8
encoding//src/uds/models.py=utf-8 encoding//src/uds/models.py=utf-8
encoding//src/uds/osmanagers/LinuxOsManager/LinuxOsManager.py=utf-8 encoding//src/uds/osmanagers/LinuxOsManager/LinuxOsManager.py=utf-8
encoding//src/uds/osmanagers/LinuxOsManager/__init__.py=utf-8 encoding//src/uds/osmanagers/LinuxOsManager/__init__.py=utf-8
encoding//src/uds/osmanagers/NoneOsManager/Manager.py=utf-8
encoding//src/uds/osmanagers/NoneOsManager/__init__.py=utf-8
encoding//src/uds/osmanagers/WindowsOsManager/WinDomainOsManager.py=utf-8 encoding//src/uds/osmanagers/WindowsOsManager/WinDomainOsManager.py=utf-8
encoding//src/uds/osmanagers/WindowsOsManager/WindowsOsManager.py=utf-8 encoding//src/uds/osmanagers/WindowsOsManager/WindowsOsManager.py=utf-8
encoding//src/uds/osmanagers/WindowsOsManager/__init__.py=utf-8 encoding//src/uds/osmanagers/WindowsOsManager/__init__.py=utf-8

View File

@ -30,16 +30,6 @@
''' '''
UDS authentication related interfaces and classes UDS authentication related interfaces and classes
From 1.0 onwards, the refactoring of UDS has started.
We can access the base service interfaces the old method, or recommended
and easier use the new one, that is "from uds.core.services import ..."
The new valid names for classes are:
I think this is an easier to use and understand way of accessing this classes
.. moduleauthor:: Adolfo Gómez, dkmaster at dkmon dot com .. moduleauthor:: Adolfo Gómez, dkmaster at dkmon dot com
''' '''
from BaseAuthenticator import Authenticator from BaseAuthenticator import Authenticator

View File

@ -37,7 +37,7 @@ from uds.core import Module
STORAGE_KEY = 'osmk' STORAGE_KEY = 'osmk'
class BaseOSManager(Module): class OSManager(Module):
''' '''
An OS Manager is responsible for communication the service the different actions to take (i.e. adding a windows machine to a domain) An OS Manager is responsible for communication the service the different actions to take (i.e. adding a windows machine to a domain)
The Service (i.e. virtual machine) communicates with the OSManager via a published web method, that must include the unique ID. The Service (i.e. virtual machine) communicates with the OSManager via a published web method, that must include the unique ID.
@ -56,7 +56,7 @@ class BaseOSManager(Module):
processUnusedMachines = False processUnusedMachines = False
def __init__(self,environment, values): def __init__(self,environment, values):
super(BaseOSManager, self).__init__(environment, values) super(OSManager, self).__init__(environment, values)
self.initialize(values) self.initialize(values)
def initialize(self, values): def initialize(self, values):

View File

@ -0,0 +1,43 @@
# -*- 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.
'''
UDS os managers related interfaces and classes
.. moduleauthor:: Adolfo Gómez, dkmaster at dkmon dot com
'''
from BaseOsManager import OSManager
def factory():
'''
Returns factory for register/access to authenticators
'''
from OSManagersFactory import OSManagersFactory
return OSManagersFactory.factory()

View File

@ -124,7 +124,7 @@ class UserDeployment(Environmentable, Serializable):
kwargs: List of arguments that will receive: kwargs: List of arguments that will receive:
service: Parent service (derived from Service) of this deployment (this is an instance, not database object) service: Parent service (derived from Service) of this deployment (this is an instance, not database object)
publication: Parent publication (derived from Publication) of this deployment (optional)(this is an instance, not database object) publication: Parent publication (derived from Publication) of this deployment (optional)(this is an instance, not database object)
osmanager: Parent osmanager (derived from BaseOsManager) of this deployment (optional)(this is an instance, not database object) osmanager: Parent osmanager (derived from :py:class:`uds.core.osmanagersOSManager`) of this deployment (optional)(this is an instance, not database object)
dbservice: Database object for this service dbservice: Database object for this service
''' '''
Environmentable.__init__(self, environment) Environmentable.__init__(self, environment)

View File

@ -33,7 +33,6 @@
from django.db import models from django.db import models
from django.db.models import signals from django.db.models import signals
from uds.core.osmanagers.OSManagersFactory import OSManagersFactory
from uds.core.jobs.JobsFactory import JobsFactory from uds.core.jobs.JobsFactory import JobsFactory
from uds.core.Environment import Environment from uds.core.Environment import Environment
from uds.core.util.db.LockingManager import LockingManager from uds.core.util.db.LockingManager import LockingManager
@ -296,8 +295,9 @@ class OSManager(models.Model):
:note: We only need to get info from this, not access specific data (class specific info) :note: We only need to get info from this, not access specific data (class specific info)
''' '''
# We only need to get info from this, not access specific data (class specific info # We only need to get info from this, not access specific data (class specific info)
return OSManagersFactory.factory().lookup(self.data_type) from uds.core import osmanagers
return osmanagers.factory().lookup(self.data_type)
def __unicode__(self): def __unicode__(self):

View File

@ -33,14 +33,15 @@
from django.utils.translation import ugettext_noop as _ from django.utils.translation import ugettext_noop as _
from uds.core.ui.UserInterface import gui from uds.core.ui.UserInterface import gui
from uds.core.osmanagers.BaseOsManager import BaseOSManager, State from uds.core import osmanagers
from uds.core.util.State import State
import logging import logging
from uds.core.managers.UserServiceManager import UserServiceManager from uds.core.managers.UserServiceManager import UserServiceManager
logger = logging.getLogger(__name__) logger = logging.getLogger(__name__)
class LinuxOsManager(BaseOSManager): class LinuxOsManager(osmanagers.OSManager):
typeName = _('Linux OS Manager') typeName = _('Linux OS Manager')
typeType = 'LinuxManager' typeType = 'LinuxManager'
typeDescription = _('Os Manager to control linux virtual machines (basically renames machine and notify state)') typeDescription = _('Os Manager to control linux virtual machines (basically renames machine and notify state)')

View File

@ -1,72 +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.
'''
@author: Adolfo Gómez, dkmaster at dkmon dot com
'''
from django.utils.translation import ugettext_noop as _
from uds.core.ui.UserInterface import gui
from uds.core.osmanagers.BaseOsManager import BaseOSManager, State
import logging
logger = logging.getLogger(__name__)
class NoneOSManager(BaseOSManager):
typeName = _('None OS Manager')
typeType = 'NoneOSManager'
typeDescription = _('Os Manager with no actions')
iconFile = 'osmanager.png'
def __init__(self,environment, values):
super(NoneOSManager, self).__init__(environment, values)
def process(self,service,msg, data):
logger.info("Invoked NoneOsManager for {0} with params: {1}, {2}".format(service, msg, data))
return "noneos"
def checkState(self,service):
logger.debug('Checking state for service {0}'.format(service))
return State.FINISHED
def marshal(self):
'''
Serializes the os manager data so we can store it in database
'''
return str.join( '\t', [ 'v1' ] )
def unmarshal(self, str):
data = str.split('\t')
if data[0] == 'v1':
pass
def valuesDict(self):
return {}

View File

@ -1,38 +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.
'''
@author: Adolfo Gómez, dkmaster at dkmon dot com
'''
from uds.core.osmanagers.OSManagersFactory import OSManagersFactory
from uds.osmanagers.NoneOsManager.Manager import NoneOSManager
# This os manager exists for testing purposes only, do not register nor use it on a production environment
#OSManagersFactory.factory().insert(NoneOSManager)

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.2 KiB

View File

@ -12,7 +12,7 @@
from django.utils.translation import ugettext_noop as _ from django.utils.translation import ugettext_noop as _
from uds.core.ui.UserInterface import gui from uds.core.ui.UserInterface import gui
from uds.core.managers.CryptoManager import CryptoManager from uds.core.managers.CryptoManager import CryptoManager
from uds.core.osmanagers.BaseOsManager import BaseOSManager, State from uds.core import osmanagers
from WindowsOsManager import WindowsOsManager, scrambleMsg from WindowsOsManager import WindowsOsManager, scrambleMsg
import logging import logging
@ -37,11 +37,11 @@ class WinDomainOsManager(WindowsOsManager):
super(WinDomainOsManager, self).__init__(environment, values) super(WinDomainOsManager, self).__init__(environment, values)
if values != None: if values != None:
if values['domain'] == '': if values['domain'] == '':
raise BaseOSManager.ValidationException(_('Must provide a domain!!!')) raise osmanagers.OSManager.ValidationException(_('Must provide a domain!!!'))
if values['account'] == '': if values['account'] == '':
raise BaseOSManager.ValidationException(_('Must provide an account to add machines to domain!!!')) raise osmanagers.OSManager.ValidationException(_('Must provide an account to add machines to domain!!!'))
if values['password'] == '': if values['password'] == '':
raise BaseOSManager.ValidationException(_('Must provide a password for the account!!!')) raise osmanagers.OSManager.ValidationException(_('Must provide a password for the account!!!'))
self._domain = values['domain'] self._domain = values['domain']
self._ou = values['ou'] self._ou = values['ou']
self._account = values['account'] self._account = values['account']

View File

@ -11,8 +11,9 @@
from django.utils.translation import ugettext_noop as _ from django.utils.translation import ugettext_noop as _
from uds.core.ui.UserInterface import gui from uds.core.ui.UserInterface import gui
from uds.core.osmanagers.BaseOsManager import BaseOSManager, State from uds.core import osmanagers
from uds.core.managers.UserServiceManager import UserServiceManager from uds.core.managers.UserServiceManager import UserServiceManager
from uds.core.util.State import State
import logging import logging
@ -31,7 +32,7 @@ def scrambleMsg(data):
return "".join(res).encode('hex') return "".join(res).encode('hex')
class WindowsOsManager(BaseOSManager): class WindowsOsManager(osmanagers.OSManager):
typeName = _('Windows Basic OS Manager') typeName = _('Windows Basic OS Manager')
typeType = 'WindowsManager' typeType = 'WindowsManager'
typeDescription = _('Os Manager to control windows machines without domain. (Basically renames machine)') typeDescription = _('Os Manager to control windows machines without domain. (Basically renames machine)')
@ -48,9 +49,9 @@ class WindowsOsManager(BaseOSManager):
try: try:
len = int(len) len = int(len)
except Exception: except Exception:
raise BaseOSManager.ValidationException(_('Length must be numeric!!')) raise osmanagers.OSManager.ValidationException(_('Length must be numeric!!'))
if len > 6 or len < 1: if len > 6 or len < 1:
raise BaseOSManager.ValidationException(_('Length must be betwen 1 and six')) raise osmanagers.OSManager.ValidationException(_('Length must be betwen 1 and six'))
return len return len
def __init__(self,environment, values): def __init__(self,environment, values):

View File

@ -28,14 +28,36 @@
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
''' '''
@author: Adolfo Gómez, dkmaster at dkmon dot com OS Manager modules for uds are contained inside this package.
To create a new OS manager module, you will need to follow this steps:
1.- Create the os manager module, probably based on an existing one
2.- Insert the module package as child of this package
3.- Import the class of your os manager module at __init__. For example::
from OSManager import SimpleOSManager
4.- Done. At Server restart, the module will be recognized, loaded and treated
The registration of modules is done locating subclases of :py:class:`uds.core.auths.Authentication`
.. moduleauthor:: Adolfo Gómez, dkmaster at dkmon dot com
''' '''
def __init__():
'''
This imports all packages that are descendant of this package, and, after that,
it register all subclases of service provider as
'''
import os.path, pkgutil import os.path, pkgutil
import sys import sys
import uds.core from uds.core import osmanagers
# Dynamically import children of this package. The __init__.py files must register, if needed, inside ServiceProviderFactory # Dinamycally import children of this package. The __init__.py files must register, if needed, inside ServiceProviderFactory
pkgpath = os.path.dirname(sys.modules[__name__].__file__) pkgpath = os.path.dirname(sys.modules[__name__].__file__)
for _, name, _ in pkgutil.iter_modules([pkgpath]): for _, name, _ in pkgutil.iter_modules([pkgpath]):
__import__(name, globals(), locals(), [], -1) __import__(name, globals(), locals(), [], -1)
p = osmanagers.OSManager
# This is marked as error in IDE, but it's not (__subclasses__)
for cls in p.__subclasses__():
osmanagers.factory().insert(cls)
__init__()

View File

@ -28,12 +28,12 @@
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
''' '''
Authentication modules for uds are contained inside this package. Service modules for uds are contained inside this package.
To create a new authentication module, you will need to follow this steps: To create a new service module, you will need to follow this steps:
1.- Create the authentication module, probably based on an existing one 1.- Create the service module, probably based on an existing one
2.- Insert the module package as child of this package 2.- Insert the module package as child of this package
3.- Import the class of your authentication module at __init__. For example:: 3.- Import the class of your service module at __init__. For example::
from Authenticator import SimpleAthenticator from Service import SimpleService
4.- Done. At Server restart, the module will be recognized, loaded and treated 4.- Done. At Server restart, the module will be recognized, loaded and treated
The registration of modules is done locating subclases of :py:class:`uds.core.auths.Authentication` The registration of modules is done locating subclases of :py:class:`uds.core.auths.Authentication`

View File

@ -31,14 +31,12 @@
@author: Adolfo Gómez, dkmaster at dkmon dot com @author: Adolfo Gómez, dkmaster at dkmon dot com
''' '''
from uds.core.transports.TransportsFactory import TransportsFactory
from uds.core.managers.UserPrefsManager import UserPrefsManager, CommonPrefs from uds.core.managers.UserPrefsManager import UserPrefsManager, CommonPrefs
from uds.core.managers.DownloadsManager import DownloadsManager from uds.core.managers.DownloadsManager import DownloadsManager
from NXTransport import NXTransport from NXTransport import NXTransport
from django.utils.translation import ugettext_noop as _ from django.utils.translation import ugettext_noop as _
import os.path, sys import os.path, sys
TransportsFactory.factory().insert(NXTransport)
UserPrefsManager.manager().registerPrefs('nx', _('NX Protocol'), UserPrefsManager.manager().registerPrefs('nx', _('NX Protocol'),
[ [
CommonPrefs.screenSizePref CommonPrefs.screenSizePref

View File

@ -9,14 +9,11 @@
@author: Adolfo Gómez, dkmaster at dkmon dot com @author: Adolfo Gómez, dkmaster at dkmon dot com
''' '''
from uds.core.transports.TransportsFactory import TransportsFactory
from uds.core.managers.UserPrefsManager import UserPrefsManager, CommonPrefs from uds.core.managers.UserPrefsManager import UserPrefsManager, CommonPrefs
from uds.transports.RDP.RDPTransport import RDPTransport from uds.transports.RDP.RDPTransport import RDPTransport
from uds.transports.RDP.TSRDPTransport import TSRDPTransport from uds.transports.RDP.TSRDPTransport import TSRDPTransport
from django.utils.translation import ugettext_noop as _ from django.utils.translation import ugettext_noop as _
TransportsFactory.factory().insert(RDPTransport)
TransportsFactory.factory().insert(TSRDPTransport)
UserPrefsManager.manager().registerPrefs('rdp', _('Remote Desktop Protocol'), UserPrefsManager.manager().registerPrefs('rdp', _('Remote Desktop Protocol'),
[ [
CommonPrefs.screenSizePref, CommonPrefs.screenSizePref,

View File

@ -31,14 +31,12 @@
@author: Adolfo Gómez, dkmaster at dkmon dot com @author: Adolfo Gómez, dkmaster at dkmon dot com
''' '''
from uds.core.transports.TransportsFactory import TransportsFactory
from uds.core.managers.UserPrefsManager import UserPrefsManager, CommonPrefs from uds.core.managers.UserPrefsManager import UserPrefsManager, CommonPrefs
from uds.core.managers.DownloadsManager import DownloadsManager from uds.core.managers.DownloadsManager import DownloadsManager
from TSNXTransport import TSNXTransport from TSNXTransport import TSNXTransport
from django.utils.translation import ugettext_noop as _ from django.utils.translation import ugettext_noop as _
import os.path, sys import os.path, sys
TransportsFactory.factory().insert(TSNXTransport)
UserPrefsManager.manager().registerPrefs('nx', _('NX Protocol'), UserPrefsManager.manager().registerPrefs('nx', _('NX Protocol'),
[ [
CommonPrefs.screenSizePref CommonPrefs.screenSizePref

View File

@ -40,7 +40,7 @@ from ..util.Helpers import dictFromData
from ..auths.AdminAuth import needs_credentials from ..auths.AdminAuth import needs_credentials
from uds.core.Environment import Environment from uds.core.Environment import Environment
import logging import logging
from uds.core.osmanagers.BaseOsManager import BaseOSManager from uds.core import osmanagers
logger = logging.getLogger(__name__) logger = logging.getLogger(__name__)
@ -108,7 +108,7 @@ def createOSManager(credentials, type, data):
sp = OSManager.objects.create(name = dict['name'], comments = dict['comments'], data_type = type) sp = OSManager.objects.create(name = dict['name'], comments = dict['comments'], data_type = type)
sp.data = sp.getInstance(dict).serialize() sp.data = sp.getInstance(dict).serialize()
sp.save() sp.save()
except BaseOSManager.ValidationException, e: except osmanagers.OSManager.ValidationException, e:
sp.delete() sp.delete()
raise ValidationException(str(e)) raise ValidationException(str(e))
except IntegrityError: # Must be exception at creation except IntegrityError: # Must be exception at creation