mirror of
https://github.com/dkmstr/openuds.git
synced 2025-01-27 14:03:53 +03:00
Creating a couple of "test services" so we can make some automated tests
This commit is contained in:
parent
01f9a1f9cd
commit
909ef91181
@ -171,7 +171,7 @@ class OSManager(Module):
|
||||
Helper method that informs if the os manager transforms the username and/or the password.
|
||||
This is used from ServicePool
|
||||
"""
|
||||
return hash(cls.processUserPassword) != hash(OSManager.processUserPassword)
|
||||
return cls.processUserPassword != OSManager.processUserPassword
|
||||
|
||||
def processUserPassword(
|
||||
self, userService: 'UserService', username: str, password: str
|
||||
|
@ -31,6 +31,8 @@
|
||||
.. moduleauthor:: Adolfo Gómez, dkmaster at dkmon dot com
|
||||
"""
|
||||
import base64
|
||||
import pickle
|
||||
import gzip
|
||||
import typing
|
||||
|
||||
|
||||
@ -45,6 +47,11 @@ class Serializable:
|
||||
"""
|
||||
__slots__ = ()
|
||||
|
||||
# Note:
|
||||
# We can include a "data" member variable in the class
|
||||
# If found, and has __dict__, then we will use it
|
||||
# on marshal and unmarshal methods
|
||||
|
||||
def __init__(self):
|
||||
pass
|
||||
|
||||
@ -55,9 +62,17 @@ class Serializable:
|
||||
The system will use in fact 'seralize' and 'deserialize' methods, but theese are
|
||||
only suitable methods to "codify" serialized values
|
||||
|
||||
:note: This method must be overridden
|
||||
:note: This method can be overriden.
|
||||
:note: if you provide a "data" member variable, and it has __dict__, then it will be used
|
||||
to marshal that data variable
|
||||
"""
|
||||
raise NotImplementedError('Base marshaler called!!!')
|
||||
# Default implementation will look for a member variable called "data"
|
||||
# This is an struct, and will be pickled by default
|
||||
|
||||
if hasattr(self, 'data') and hasattr(getattr(self, 'data'), '__dict__'):
|
||||
return pickle.dumps(getattr(self, 'data'), protocol=pickle.HIGHEST_PROTOCOL) # type: ignore
|
||||
|
||||
raise NotImplementedError('You must override the marshal method or provide a data member')
|
||||
|
||||
def unmarshal(self, data: bytes) -> None:
|
||||
"""
|
||||
@ -72,9 +87,15 @@ class Serializable:
|
||||
Args:
|
||||
data : String readed from persistent storage to deseralilize
|
||||
|
||||
:note: This method must be overridden
|
||||
:note: This method can be overriden.
|
||||
:note: if you provide a "data" member variable, and it has __dict__, then it will be used
|
||||
to unmarshal that data variable
|
||||
"""
|
||||
raise NotImplementedError('Base unmarshaler called!!!')
|
||||
if hasattr(self, 'data') and hasattr(getattr(self, 'data'), '__dict__'):
|
||||
setattr(self, 'data', pickle.loads(data))
|
||||
return
|
||||
|
||||
raise NotImplementedError('You must override the unmarshal method or provide a data member')
|
||||
|
||||
def serialize(self) -> str:
|
||||
"""
|
||||
|
@ -1,7 +1,7 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
#
|
||||
# Copyright (c) 2012-2019 Virtual Cable S.L.U.
|
||||
# Copyright (c) 2012-2022 Virtual Cable S.L.U.
|
||||
# All rights reserved.
|
||||
#
|
||||
# Redistribution and use in source and binary forms, with or without modification,
|
||||
@ -47,9 +47,7 @@ if typing.TYPE_CHECKING:
|
||||
from uds.core.util.unique_gid_generator import UniqueGIDGenerator
|
||||
|
||||
|
||||
class UserDeployment(
|
||||
Environmentable, Serializable
|
||||
): # pylint: disable=too-many-public-methods
|
||||
class UserDeployment(Environmentable, Serializable):
|
||||
"""
|
||||
Interface for deployed services.
|
||||
|
||||
|
36
server/src/uds/services/Test/__init__.py
Normal file
36
server/src/uds/services/Test/__init__.py
Normal file
@ -0,0 +1,36 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
#
|
||||
# Copyright (c) 2022 Virtual Cable S.L.U.
|
||||
# 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.U. 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.
|
||||
|
||||
"""
|
||||
Simple "testing" provider.
|
||||
|
||||
This package provides a simple test provider, suitable for automated tests.
|
||||
"""
|
||||
|
||||
from .provider import Provider
|
141
server/src/uds/services/Test/deployment_one.py
Normal file
141
server/src/uds/services/Test/deployment_one.py
Normal file
@ -0,0 +1,141 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
#
|
||||
# Copyright (c) 20122 Virtual Cable S.L.U.
|
||||
# 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.U. 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 dataclasses
|
||||
import typing
|
||||
|
||||
from uds.core import services
|
||||
from uds.core.util.state import State
|
||||
|
||||
from . import service
|
||||
|
||||
# Not imported at runtime, just for type checking
|
||||
if typing.TYPE_CHECKING:
|
||||
from uds import models
|
||||
from .service import ServiceTestNoCache
|
||||
from .publication import TestPublication
|
||||
|
||||
logger = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class TestUserDeploymentNoCache(services.UserDeployment):
|
||||
"""
|
||||
Simple testing deployment, no cache
|
||||
"""
|
||||
|
||||
@dataclasses.dataclass
|
||||
class Data:
|
||||
"""
|
||||
This is the data we will store in the storage
|
||||
"""
|
||||
|
||||
count: int = -1
|
||||
ready: bool = False
|
||||
name: str = ''
|
||||
ip: str = ''
|
||||
mac: str = ''
|
||||
|
||||
data: Data = dataclasses.field(default_factory=Data)
|
||||
|
||||
# : Recheck every five seconds by default (for task methods)
|
||||
suggestedTime = 5
|
||||
|
||||
def service(self) -> 'ServiceTestNoCache':
|
||||
return typing.cast('ServiceTestNoCache', super().service())
|
||||
|
||||
def getName(self) -> str:
|
||||
if not self.data.name:
|
||||
self.data.name = self.nameGenerator().get(self.service().getBaseName(), 3)
|
||||
|
||||
logger.info('Getting name of deployment %s', self.data)
|
||||
|
||||
return self.data.name
|
||||
|
||||
def setIp(self, ip: str) -> None:
|
||||
logger.info('Setting ip of deployment %s to %s', self.data, ip)
|
||||
self.data.ip = ip
|
||||
|
||||
def getUniqueId(self) -> str:
|
||||
logger.info('Getting unique id of deployment %s', self.data)
|
||||
if not self.data.mac:
|
||||
self.data.mac = self.macGenerator().get(
|
||||
'00:00:00:00:00:00-00:FF:FF:FF:FF:FF'
|
||||
)
|
||||
return self.data.mac
|
||||
|
||||
def getIp(self) -> str:
|
||||
logger.info('Getting ip of deployment %s', self.data)
|
||||
ip = typing.cast(str, self.storage.readData('ip'))
|
||||
if ip is None:
|
||||
ip = '8.6.4.2' # Sample IP for testing purposses only
|
||||
return ip
|
||||
|
||||
def setReady(self) -> str:
|
||||
logger.info('Setting ready %s', self.data)
|
||||
self.data.ready = True
|
||||
return State.FINISHED
|
||||
|
||||
def deployForUser(self, user: 'models.User') -> str:
|
||||
logger.info('Deploying for user %s %s', user, self.data)
|
||||
self.data.count = 10
|
||||
return State.RUNNING
|
||||
|
||||
def checkState(self) -> str:
|
||||
logger.info('Checking state of deployment %s', self.data)
|
||||
if self.data.count <= 0:
|
||||
return State.FINISHED
|
||||
|
||||
self.data.count -= 1
|
||||
return State.RUNNING
|
||||
|
||||
def finish(self) -> None:
|
||||
logger.info('Finishing deployment %s', self.data)
|
||||
self.data.count = -1
|
||||
|
||||
def userLoggedIn(self, username: str) -> None:
|
||||
logger.info('User %s has logged in', username)
|
||||
|
||||
def userLoggedOut(self, username) -> None:
|
||||
logger.info('User %s has logged out', username)
|
||||
|
||||
def reasonOfError(self) -> str:
|
||||
return 'No error'
|
||||
|
||||
def destroy(self) -> str:
|
||||
logger.info('Destroying deployment %s', self.data)
|
||||
self.data.count = -1
|
||||
return State.FINISHED
|
||||
|
||||
def cancel(self) -> str:
|
||||
return self.destroy()
|
||||
|
445
server/src/uds/services/Test/deployment_two.py
Normal file
445
server/src/uds/services/Test/deployment_two.py
Normal file
@ -0,0 +1,445 @@
|
||||
# -*- 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 codecs
|
||||
import logging
|
||||
import typing
|
||||
|
||||
from uds.core import services
|
||||
from uds.core.util.state import State
|
||||
|
||||
# Not imported at runtime, just for type checking
|
||||
if typing.TYPE_CHECKING:
|
||||
from uds import models
|
||||
from .service import ServiceTestNoCache
|
||||
from .publication import TestPublication
|
||||
|
||||
logger = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class TestUserDeploymentCache(services.UserDeployment):
|
||||
"""
|
||||
This class generates the user consumable elements of the service tree.
|
||||
|
||||
This is almost the same as SampleUserDeploymentOne, but differs that this one
|
||||
uses the publication to get data from it, in a very basic way.
|
||||
|
||||
After creating at administration interface an Deployed Service, UDS will
|
||||
create consumable services for users using UserDeployment class as
|
||||
provider of this elements.
|
||||
|
||||
At class instantiation, this will receive an environment with"generator",
|
||||
that are classes that provides a way to generate unique items.
|
||||
|
||||
The generators provided right now are 'mac' and 'name'. To get more info
|
||||
about this, look at py:class:`uds.core.util.unique_mac_generator.UniqueNameGenerator`
|
||||
and py:class:`uds.core.util.unique_name_generator.UniqueNameGenerator`
|
||||
|
||||
As sample also of environment storage usage, wi will use here the provider
|
||||
storage to keep all our needed info, leaving marshal and unmarshal (needed
|
||||
by Serializable classes, like this) empty (that is, returns '' first and does
|
||||
nothing the second one)
|
||||
|
||||
Also Remember, if you don't include this class as the deployedType of the
|
||||
SampleServiceTwo, or whenever you try to access a service of SampleServiceTwo,
|
||||
you will get an exception that says that you haven't included the deployedType.
|
||||
"""
|
||||
|
||||
# : Recheck every five seconds by default (for task methods)
|
||||
suggestedTime = 2
|
||||
|
||||
_name: str
|
||||
_ip: str
|
||||
_mac: str
|
||||
_error: str
|
||||
_count: int
|
||||
|
||||
# Utility overrides for type checking...
|
||||
def service(self) -> 'ServiceTestNoCache':
|
||||
return typing.cast('ServiceOne', super().service())
|
||||
|
||||
def publication(self) -> 'TestPublication':
|
||||
pub = super().publication()
|
||||
if pub is None:
|
||||
raise Exception('No publication for this element!')
|
||||
return typing.cast('SamplePublication', pub)
|
||||
|
||||
def initialize(self) -> None:
|
||||
"""
|
||||
Initialize default attributes values here. We can do whatever we like,
|
||||
but for this sample this is just right...
|
||||
"""
|
||||
self._name = ''
|
||||
self._ip = ''
|
||||
self._mac = ''
|
||||
self._error = ''
|
||||
self._count = 0
|
||||
|
||||
# Serializable needed methods
|
||||
def marshal(self) -> bytes:
|
||||
"""
|
||||
Marshal own data, in this sample we will marshal internal needed
|
||||
attributes.
|
||||
|
||||
In this case, the data will be store with the database record. To
|
||||
minimize database storage usage, we will "zip" data before returning it.
|
||||
Anyway, we should keep this data as low as possible, we also have an
|
||||
storage for loading larger data.
|
||||
|
||||
:note: It's a good idea when providing marshalers, to store a 'version'
|
||||
beside the values, so we can, at a later stage, treat with old
|
||||
data for current modules.
|
||||
"""
|
||||
data = '\t'.join(
|
||||
['v1', self._name, self._ip, self._mac, self._error, str(self._count)]
|
||||
)
|
||||
return codecs.encode(data.encode(), encoding='zip') # type: ignore
|
||||
|
||||
def unmarshal(self, data: bytes) -> None:
|
||||
"""
|
||||
We unmarshal the content.
|
||||
"""
|
||||
values: typing.List[str] = codecs.decode(data, 'zip').decode().split('\t') # type: ignore
|
||||
# Data Version check
|
||||
# If we include some new data at some point in a future, we can
|
||||
# add "default" values at v1 check, and load new values at 'v2' check.
|
||||
if values[0] == 'v1':
|
||||
self._name, self._ip, self._mac, self._error, count = values[1:]
|
||||
self._count = int(count)
|
||||
|
||||
def getName(self) -> str:
|
||||
"""
|
||||
We override this to return a name to display. Default implementation
|
||||
(in base class), returns getUniqueIde() value
|
||||
This name will help user to identify elements, and is only used
|
||||
at administration interface.
|
||||
|
||||
We will use here the environment name provided generator to generate
|
||||
a name for this element.
|
||||
|
||||
The namaGenerator need two params, the base name and a length for a
|
||||
numeric incremental part for generating unique names. This are unique for
|
||||
all UDS names generations, that is, UDS will not generate this name again
|
||||
until this name is freed, or object is removed, what makes its environment
|
||||
to also get removed, that makes all unique ids (names and macs right now)
|
||||
to also get released.
|
||||
|
||||
Every time get method of a generator gets called, the generator creates
|
||||
a new unique name, so we keep the first generated name cached and don't
|
||||
generate more names. (Generator are simple utility classes)
|
||||
"""
|
||||
if self._name == '':
|
||||
self._name = self.nameGenerator().get(self.publication().getBaseName(), 3)
|
||||
# self._name will be stored when object is marshaled
|
||||
return self._name
|
||||
|
||||
def setIp(self, ip: str) -> None:
|
||||
"""
|
||||
In our case, there is no OS manager associated with this, so this method
|
||||
will never get called, but we put here as sample.
|
||||
|
||||
Whenever an os manager actor notifies the broker the state of the service
|
||||
(mainly machines), the implementation of that os manager can (an probably will)
|
||||
need to notify the IP of the deployed service. Remember that UDS treats with
|
||||
IP services, so will probable needed in every service that you will create.
|
||||
:note: This IP is the IP of the "consumed service", so the transport can
|
||||
access it.
|
||||
"""
|
||||
self._ip = ip
|
||||
|
||||
def getUniqueId(self) -> str:
|
||||
"""
|
||||
Return and unique identifier for this service.
|
||||
In our case, we will generate a mac name, that can be also as sample
|
||||
of 'mac' generator use, and probably will get used something like this
|
||||
at some services.
|
||||
|
||||
The get method of a mac generator takes one param, that is the mac range
|
||||
to use to get an unused mac.
|
||||
|
||||
The mac generated is not used by anyone, it will not depend on
|
||||
the range, the generator will take care that this mac is unique
|
||||
and in the range provided, or it will return None. The ranges
|
||||
are wide enough to ensure that we always will get a mac address
|
||||
in this case, but if this is not your case, take into account that
|
||||
None is a possible return value, and in that case, you should return an
|
||||
invalid id right now. Every time a task method is invoked, the core
|
||||
will try to update the value of the unique id using this method, so
|
||||
that id can change with time. (In fact, it's not unique at database level,
|
||||
it's unique in the sense that you must return an unique id that can, for
|
||||
example, be used by os managers to identify this element).
|
||||
|
||||
:note: Normally, getting out of macs in the mac pool is a bad thing... :-)
|
||||
"""
|
||||
if self._mac == '':
|
||||
self._mac = self.macGenerator().get('00:00:00:00:00:00-00:FF:FF:FF:FF:FF')
|
||||
return self._mac
|
||||
|
||||
def getIp(self) -> str:
|
||||
"""
|
||||
We need to implement this method, so we can return the IP for transports
|
||||
use. If no IP is known for this service, this must return None
|
||||
|
||||
If our sample do not returns an IP, IP transport will never work with
|
||||
this service. Remember in real cases to return a valid IP address if
|
||||
the service is accesible and you alredy know that (for example, because
|
||||
the IP has been assigend via setIp by an os manager) or because
|
||||
you get it for some other method.
|
||||
|
||||
Storage returns None if key is not stored.
|
||||
|
||||
:note: Keeping the IP address is responsibility of the User Deployment.
|
||||
Every time the core needs to provide the service to the user, or
|
||||
show the IP to the administrator, this method will get called
|
||||
|
||||
"""
|
||||
if self._ip == '':
|
||||
return '192.168.0.34' # Sample IP for testing purposes only
|
||||
return self._ip
|
||||
|
||||
def setReady(self) -> str:
|
||||
"""
|
||||
This is a task method. As that, the excepted return values are
|
||||
State values RUNNING, FINISHED or ERROR.
|
||||
|
||||
The method is invoked whenever a machine is provided to an user, right
|
||||
before presenting it (via transport rendering) to the user.
|
||||
|
||||
This method exist for this kind of situations (i will explain it with a
|
||||
sample)
|
||||
|
||||
Imagine a Service tree (Provider, Service, ...) for virtual machines.
|
||||
This machines will get created by the UserDeployment implementation, but,
|
||||
at some time, the machine can be put at in an state (suspend, shut down)
|
||||
that will make the transport impossible to connect with it.
|
||||
|
||||
This method, in this case, will check the state of the machine, and if
|
||||
it is "ready", that is, powered on and accessible, it will return
|
||||
"State.FINISHED". If the machine is not accessible (has been erased, for
|
||||
example), it will return "State.ERROR" and store a reason of error so UDS
|
||||
can ask for it and present this information to the Administrator.
|
||||
|
||||
If the machine powered off, or suspended, or any other state that is not
|
||||
directly usable but can be put in an usable state, it will return
|
||||
"State.RUNNING", and core will use checkState to see when the operation
|
||||
has finished.
|
||||
|
||||
I hope this sample is enough to explain the use of this method..
|
||||
"""
|
||||
|
||||
# In our case, the service is always ready
|
||||
return State.FINISHED
|
||||
|
||||
def deployForUser(self, user: 'models.User') -> str:
|
||||
"""
|
||||
Deploys an service instance for an user.
|
||||
|
||||
This is a task method. As that, the excepted return values are
|
||||
State values RUNNING, FINISHED or ERROR.
|
||||
|
||||
The user parameter is not realy neded, but provided. It indicates the
|
||||
Database User Object (see py:mod:`uds.modules`) to which this deployed
|
||||
user service will be assigned to.
|
||||
|
||||
This method will get called whenever a new deployed service for an user
|
||||
is needed. This will give this class the oportunity to create
|
||||
a service that is assigned to an user.
|
||||
|
||||
The way of using this method is as follows:
|
||||
|
||||
If the service gets created in "one step", that is, before the return
|
||||
of this method, the consumable service for the user gets created, it
|
||||
will return "State.FINISH".
|
||||
If the service needs more steps (as in this case), we will return
|
||||
"State.RUNNING", and if it has an error, it wil return "State.ERROR" and
|
||||
store an error string so administration interface can show it.
|
||||
|
||||
We do not use user for anything, as in most cases will be.
|
||||
"""
|
||||
import random
|
||||
|
||||
self._count = 0
|
||||
|
||||
# random fail
|
||||
if random.randint(0, 9) == 9:
|
||||
# Note that we can mark this string as translatable, and return
|
||||
# it translated at reasonOfError method
|
||||
self._error = 'Random error at deployForUser :-)'
|
||||
return State.ERROR
|
||||
|
||||
return State.RUNNING
|
||||
|
||||
def deployForCache(self, cacheLevel: int) -> str:
|
||||
"""
|
||||
Deploys a user deployment as cache.
|
||||
|
||||
This is a task method. As that, the expected return values are
|
||||
State values RUNNING, FINISHED or ERROR.
|
||||
|
||||
In our sample, this will do exactly the same as deploy for user,
|
||||
except that it will never will give an error.
|
||||
|
||||
See deployForUser for a description of what this method should do.
|
||||
|
||||
:note: deployForCache is invoked whenever a new cache element is needed
|
||||
for an specific user deployment. It will also indicate for what
|
||||
cache level (L1, L2) is the deployment
|
||||
"""
|
||||
self._count = 0
|
||||
return State.RUNNING
|
||||
|
||||
def checkState(self) -> str:
|
||||
"""
|
||||
Our deployForUser method will initiate the consumable service deployment,
|
||||
but will not finish it.
|
||||
|
||||
So in our sample, we will only check if a number reaches 5, and if so
|
||||
return that we have finished, else we will return that we are working
|
||||
on it.
|
||||
|
||||
One deployForUser returns State.RUNNING, this task will get called until
|
||||
checkState returns State.FINISHED.
|
||||
|
||||
Also, we will make the user deployment fail one of every 10 calls to this
|
||||
method.
|
||||
|
||||
Note: Destroying, canceling and deploying for cache also makes use of
|
||||
this method, so you must keep the info of that you are checking if you
|
||||
need it.
|
||||
|
||||
In our case, destroy is 1-step action so this will no get called while
|
||||
destroying, and cancel will simply invoke destroy. Cache deployment is
|
||||
exactly as user deployment, except that the core will not assign it to
|
||||
anyone, and cache moving operations is
|
||||
"""
|
||||
import random
|
||||
|
||||
self._count += 1
|
||||
# Count is always a valid value, because this method will never get
|
||||
# called before deployForUser, deployForCache, destroy or cancel.
|
||||
# In our sample, we only use checkState in case of deployForUser,
|
||||
# so at first call count will be 0.
|
||||
if self._count >= 5:
|
||||
return State.FINISHED
|
||||
|
||||
# random fail
|
||||
if random.randint(0, 9) == 9:
|
||||
self._error = 'Random error at checkState :-)'
|
||||
return State.ERROR
|
||||
|
||||
return State.RUNNING
|
||||
|
||||
def finish(self):
|
||||
"""
|
||||
Invoked when the core notices that the deployment of a service has finished.
|
||||
(No matter whether it is for cache or for an user)
|
||||
|
||||
This gives the opportunity to make something at that moment.
|
||||
|
||||
:note: You can also make these operations at checkState, this is really
|
||||
not needed, but can be provided (default implementation of base class does
|
||||
nothing)
|
||||
"""
|
||||
# We set count to 0, not needed but for sample purposes
|
||||
self._count = 0
|
||||
|
||||
def userLoggedIn(self, username: str) -> None:
|
||||
"""
|
||||
This method must be available so os managers can invoke it whenever
|
||||
an user get logged into a service.
|
||||
|
||||
Default implementation does nothing, so if you are going to do nothing,
|
||||
you don't need to implement it.
|
||||
|
||||
The responsibility of notifying it is of os manager actor, and it's
|
||||
directly invoked by os managers (right now, linux os manager and windows
|
||||
os manager)
|
||||
|
||||
The user provided is just an string, that is provided by actors.
|
||||
"""
|
||||
# We store the value at storage, but never get used, just an example
|
||||
self.storage.saveData('user', username)
|
||||
|
||||
def userLoggedOut(self, username) -> None:
|
||||
"""
|
||||
This method must be available so os managers can invoke it whenever
|
||||
an user get logged out if a service.
|
||||
|
||||
Default implementation does nothing, so if you are going to do nothing,
|
||||
you don't need to implement it.
|
||||
|
||||
The responability of notifying it is of os manager actor, and it's
|
||||
directly invoked by os managers (right now, linux os manager and windows
|
||||
os manager)
|
||||
|
||||
The user provided is just an string, that is provided by actor.
|
||||
"""
|
||||
# We do nothing more that remove the user
|
||||
self.storage.remove('user')
|
||||
|
||||
def reasonOfError(self) -> str:
|
||||
"""
|
||||
Returns the reason of the error.
|
||||
|
||||
Remember that the class is responsible of returning this whenever asked
|
||||
for it, and it will be asked everytime it's needed to be shown to the
|
||||
user (when the administation asks for it).
|
||||
|
||||
:note: Remember that you can use gettext to translate this error to
|
||||
user language whenever it is possible. (This one will get invoked
|
||||
directly from admin interface and, as so, will have translation
|
||||
environment correctly set up.
|
||||
"""
|
||||
return self._error
|
||||
|
||||
def destroy(self) -> str:
|
||||
"""
|
||||
This is a task method. As that, the excepted return values are
|
||||
State values RUNNING, FINISHED or ERROR.
|
||||
|
||||
Invoked for destroying a deployed service
|
||||
Do whatever needed here, as deleting associated data if needed (i.e. a copy of the machine, snapshots, etc...)
|
||||
@return: State.FINISHED if no more checks/steps for deployment are needed, State.RUNNING if more steps are needed (steps checked using checkState)
|
||||
"""
|
||||
return State.FINISHED
|
||||
|
||||
def cancel(self) -> str:
|
||||
"""
|
||||
This is a task method. As that, the excepted return values are
|
||||
State values RUNNING, FINISHED or ERROR.
|
||||
|
||||
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.
|
||||
"""
|
||||
return State.FINISHED
|
BIN
server/src/uds/services/Test/provider.png
Normal file
BIN
server/src/uds/services/Test/provider.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 491 B |
99
server/src/uds/services/Test/provider.py
Normal file
99
server/src/uds/services/Test/provider.py
Normal file
@ -0,0 +1,99 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
#
|
||||
# Copyright (c) 2012-2019 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.
|
||||
|
||||
"""
|
||||
Created on Jun 22, 2012
|
||||
|
||||
.. moduleauthor:: Adolfo Gómez, dkmaster at dkmon dot com
|
||||
"""
|
||||
import logging
|
||||
import random
|
||||
import string
|
||||
import typing
|
||||
|
||||
|
||||
from django.utils.translation import gettext_noop as _
|
||||
from uds.core import services
|
||||
from .service import ServiceTestNoCache, ServiceTestCache
|
||||
|
||||
# Not imported at runtime, just for type checking
|
||||
if typing.TYPE_CHECKING:
|
||||
from uds.core.environment import Environment
|
||||
|
||||
logger = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class Provider(services.ServiceProvider):
|
||||
"""
|
||||
This class represents the simple Test provider.
|
||||
|
||||
This is only intended for testing purposes, and is not a good example of
|
||||
a provider.
|
||||
|
||||
"""
|
||||
|
||||
# : What kind of services we offer, this are classes inherited from Service
|
||||
offers = [ServiceTestNoCache, ServiceTestCache]
|
||||
# : Name to show the administrator. This string will be translated BEFORE
|
||||
# : sending it to administration interface, so don't forget to
|
||||
# : mark it as _ (using gettext_noop)
|
||||
typeName = _('Testing Provider')
|
||||
# : Type used internally to identify this provider
|
||||
typeType = 'TestProvider'
|
||||
# : Description shown at administration interface for this provider
|
||||
typeDescription = _('Test (and dummy) service provider')
|
||||
# : Icon file used as icon for this provider. This string will be translated
|
||||
# : BEFORE sending it to administration interface, so don't forget to
|
||||
# : mark it as _ (using gettext_noop)
|
||||
iconFile = 'provider.png'
|
||||
|
||||
# now comes the form fields
|
||||
# There is always two fields that are requested to the admin, that are:
|
||||
# Service Name, that is a name that the admin uses to name this provider
|
||||
# Description, that is a short description that the admin gives to this provider
|
||||
# Now we are going to add a few fields that we need to use this provider
|
||||
# Remember that these are "dummy" fields, that in fact are not required
|
||||
# but used for sample purposes
|
||||
# If we don't indicate an order, the output order of fields will be
|
||||
# "random"
|
||||
|
||||
# No fields
|
||||
|
||||
@staticmethod
|
||||
def test(
|
||||
env: 'Environment', data: typing.Dict[str, str]
|
||||
) -> typing.List[typing.Any]:
|
||||
return [True, _('Nothing tested, but all went fine..')]
|
||||
|
||||
def getName(self) -> str:
|
||||
"""
|
||||
returns a random name for testing pourposes
|
||||
"""
|
||||
return '-'.join(random.choices(string.ascii_letters, k=10))
|
||||
|
121
server/src/uds/services/Test/publication.py
Normal file
121
server/src/uds/services/Test/publication.py
Normal file
@ -0,0 +1,121 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
#
|
||||
# Copyright (c) 2022 Virtual Cable S.L.U.
|
||||
# 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 random
|
||||
import string
|
||||
import logging
|
||||
import dataclasses
|
||||
import typing
|
||||
|
||||
from django.utils.translation import gettext as _
|
||||
from uds.core import services
|
||||
from uds.core.util.state import State
|
||||
|
||||
logger = logging.getLogger(__name__)
|
||||
|
||||
# Not imported at runtime, just for type checking
|
||||
if typing.TYPE_CHECKING:
|
||||
from .service import ServiceTestNoCache, ServiceTestCache
|
||||
|
||||
|
||||
class TestPublication(services.Publication):
|
||||
"""
|
||||
Simple test publication
|
||||
"""
|
||||
suggestedTime = (
|
||||
5 # : Suggested recheck time if publication is unfinished in seconds
|
||||
)
|
||||
|
||||
# Data to store
|
||||
@dataclasses.dataclass
|
||||
class Data:
|
||||
name: str = ''
|
||||
state: str = ''
|
||||
reason: str = ''
|
||||
number: int = -1
|
||||
other: str = ''
|
||||
other2: str = 'other2'
|
||||
|
||||
|
||||
data: Data = Data()
|
||||
|
||||
def initialize(self) -> None:
|
||||
"""
|
||||
This method will be invoked by default __init__ of base class, so it gives
|
||||
us the oportunity to initialize whataver we need here.
|
||||
|
||||
In our case, we setup a few attributes..
|
||||
"""
|
||||
|
||||
# We do not check anything at marshal method, so we ensure that
|
||||
# default values are correctly handled by marshal.
|
||||
self.data.name = ''.join(random.choices(string.ascii_letters, k=8))
|
||||
self.data.state = State.RUNNING
|
||||
self.data.reason = 'none'
|
||||
self.data.number = 10
|
||||
|
||||
def publish(self) -> str:
|
||||
logger.info('Publishing publication %s: %s remaining',self.data.name, self.data.number)
|
||||
self.data.number -= 1
|
||||
|
||||
if self.data.number <= 0:
|
||||
self.data.state = State.FINISHED
|
||||
return self.data.state
|
||||
|
||||
def finish(self) -> None:
|
||||
# Make simply a random string
|
||||
logger.info('Finishing publication %s', self.data.name)
|
||||
self.data.number = 0
|
||||
self.data.state = State.FINISHED
|
||||
|
||||
def reasonOfError(self) -> str:
|
||||
return self.data.reason
|
||||
|
||||
def destroy(self) -> str:
|
||||
logger.info('Destroying publication %s', self.data.name)
|
||||
return State.FINISHED
|
||||
|
||||
def cancel(self) -> str:
|
||||
logger.info('Canceling publication %s', self.data.name)
|
||||
return self.destroy()
|
||||
|
||||
# Here ends the publication needed methods.
|
||||
# Methods provided below are specific for this publication
|
||||
# and will be used by user deployments that uses this kind of publication
|
||||
|
||||
def getBaseName(self) -> str:
|
||||
"""
|
||||
This sample method (just for this sample publication), provides
|
||||
the name generater for this publication. This is just a sample, and
|
||||
this will do the work
|
||||
"""
|
||||
return self.data.name
|
BIN
server/src/uds/services/Test/service.png
Normal file
BIN
server/src/uds/services/Test/service.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 578 B |
146
server/src/uds/services/Test/service.py
Normal file
146
server/src/uds/services/Test/service.py
Normal file
@ -0,0 +1,146 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
#
|
||||
# Copyright (c) 2012-2019 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 typing
|
||||
|
||||
from django.utils.translation import gettext_noop as _
|
||||
from uds.core import services
|
||||
from uds.core.ui import gui
|
||||
|
||||
from .publication import TestPublication
|
||||
from .deployment_one import TestUserDeploymentNoCache
|
||||
from .deployment_two import TestUserDeploymentCache
|
||||
|
||||
|
||||
# Not imported at runtime, just for type checking
|
||||
if typing.TYPE_CHECKING:
|
||||
from .provider import Provider
|
||||
|
||||
logger = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class ServiceTestNoCache(services.Service):
|
||||
"""
|
||||
Basic testing service without cache and no publication OFC
|
||||
|
||||
"""
|
||||
|
||||
# : Name to show the administrator. This string will be translated BEFORE
|
||||
# : sending it to administration interface, so don't forget to
|
||||
# : mark it as _ (using gettext_noop)
|
||||
typeName = _('Testing Service no cache')
|
||||
# : Type used internally to identify this provider
|
||||
typeType = 'TestServiceNoCache'
|
||||
# : Description shown at administration interface for this provider
|
||||
typeDescription = _('Testing (and dummy) service with no cache')
|
||||
# : Icon file used as icon for this provider. This string will be translated
|
||||
# : BEFORE sending it to administration interface, so don't forget to
|
||||
# : mark it as _ (using gettext_noop)
|
||||
iconFile = 'service.png'
|
||||
|
||||
# Functional related data
|
||||
|
||||
# : If the service provides more than 1 "deployed user" (-1 = no limit,
|
||||
# : 0 = ???? (do not use it!!!), N = max number to deploy
|
||||
maxDeployed = -1
|
||||
# : If we need to generate "cache" for this service, so users can access the
|
||||
# : provided services faster. Is usesCache is True, you will need also
|
||||
# : set publicationType, do take care about that!
|
||||
usesCache = False
|
||||
# : Tooltip shown to user when this item is pointed at admin interface, none
|
||||
# : because we don't use it
|
||||
cacheTooltip = _('None')
|
||||
# : If we need to generate a "Level 2" cache for this service (i.e., L1
|
||||
# : could be running machines and L2 suspended machines)
|
||||
usesCache_L2 = False
|
||||
# : Tooltip shown to user when this item is pointed at admin interface, None
|
||||
# : also because we don't use it
|
||||
cacheTooltip_L2 = _('None')
|
||||
|
||||
# : If the service needs a s.o. manager (managers are related to agents
|
||||
# : provided by services itselfs, i.e. virtual machines with actors)
|
||||
needsManager = False
|
||||
# : If true, the system can't do an automatic assignation of a deployed user
|
||||
# : service from this service
|
||||
mustAssignManually = False
|
||||
|
||||
# : Types of publications (preparated data for deploys)
|
||||
# : In our case, we do no need a publication, so this is None
|
||||
publicationType = None
|
||||
# : Types of deploys (services in cache and/or assigned to users)
|
||||
deployedType = TestUserDeploymentNoCache
|
||||
|
||||
def parent(self) -> 'Provider':
|
||||
return typing.cast('Provider', super().parent())
|
||||
|
||||
def getName(self) -> str:
|
||||
return self.parent().getName() + '{' + self.typeName + '}'
|
||||
|
||||
def getBaseName(self) -> str:
|
||||
return self.parent().getName()
|
||||
|
||||
class ServiceTestCache(services.Service):
|
||||
"""
|
||||
A simple testging service WITH cache and publication OFC
|
||||
"""
|
||||
|
||||
typeName = _('Testing Service WITH cache')
|
||||
typeType = 'TestingServiceCache'
|
||||
typeDescription = _('Testing (and dummy) service with CACHE and PUBLICATION')
|
||||
iconFile = 'provider.png' # : We reuse provider icon here :-), it's just for testing purpuoses
|
||||
|
||||
# Functional related data
|
||||
maxDeployed = -1
|
||||
usesCache = True
|
||||
cacheTooltip = _('L1 cache for dummy elements')
|
||||
usesCache_L2 = True
|
||||
cacheTooltip_L2 = _('L2 cache for dummy elements')
|
||||
|
||||
needsManager = False
|
||||
mustAssignManually = False
|
||||
|
||||
# : Types of publications. In this case, we will include a publication
|
||||
# : type for this one
|
||||
# : Note that this is a MUST if you indicate that needPublication
|
||||
publicationType = TestPublication
|
||||
# : Types of deploys (services in cache and/or assigned to users)
|
||||
deployedType = TestUserDeploymentCache
|
||||
|
||||
def parent(self) -> 'Provider':
|
||||
return typing.cast('Provider', super().parent())
|
||||
|
||||
def getName(self) -> str:
|
||||
return self.parent().getName() + '{' + self.typeName + '}'
|
||||
|
||||
def getBaseName(self) -> str:
|
||||
return self.parent().getName()
|
@ -1,7 +1,7 @@
|
||||
|
||||
# -*- coding: utf-8 -*-
|
||||
#
|
||||
# Copyright (c) 2022 Virtual Cable S.L.
|
||||
# Copyright (c) 2022 Virtual Cable S.L.U.
|
||||
# All rights reserved.
|
||||
#
|
||||
# Redistribution and use in source and binary forms, with or without modification,
|
||||
@ -12,7 +12,7 @@
|
||||
# * 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
|
||||
# * Neither the name of Virtual Cable S.L.U. nor the names of its contributors
|
||||
# may be used to endorse or promote products derived from this software
|
||||
# without specific prior written permission.
|
||||
#
|
||||
|
Loading…
x
Reference in New Issue
Block a user