Add jobs api
This is part-1 of jobs support in Qinling. With job API, users can create a job with an existing function that can be ran in cron job's fashion, but the job definition has more capabilities than cron. Change-Id: I0cb885dd6005ba024e3816bae9b79b0d3f35d335
This commit is contained in:
parent
ebe601b8f5
commit
5baa3f1e45
137
qinling/api/controllers/v1/job.py
Normal file
137
qinling/api/controllers/v1/job.py
Normal file
@ -0,0 +1,137 @@
|
|||||||
|
# Copyright 2017 Catalyst IT Limited
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
# you may not use this file except in compliance with the License.
|
||||||
|
# You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
|
||||||
|
import croniter
|
||||||
|
import datetime
|
||||||
|
|
||||||
|
from dateutil import parser
|
||||||
|
from oslo_config import cfg
|
||||||
|
from oslo_log import log as logging
|
||||||
|
from oslo_utils import timeutils
|
||||||
|
from pecan import rest
|
||||||
|
import six
|
||||||
|
import wsmeext.pecan as wsme_pecan
|
||||||
|
|
||||||
|
from qinling.api.controllers.v1 import resources
|
||||||
|
from qinling.api.controllers.v1 import types
|
||||||
|
from qinling.db import api as db_api
|
||||||
|
from qinling import exceptions as exc
|
||||||
|
from qinling.utils.openstack import keystone as keystone_util
|
||||||
|
from qinling.utils import rest_utils
|
||||||
|
|
||||||
|
LOG = logging.getLogger(__name__)
|
||||||
|
CONF = cfg.CONF
|
||||||
|
|
||||||
|
POST_REQUIRED = set(['function_id'])
|
||||||
|
|
||||||
|
|
||||||
|
class JobsController(rest.RestController):
|
||||||
|
@rest_utils.wrap_wsme_controller_exception
|
||||||
|
@wsme_pecan.wsexpose(
|
||||||
|
resources.Job,
|
||||||
|
body=resources.Job,
|
||||||
|
status_code=201
|
||||||
|
)
|
||||||
|
def post(self, job):
|
||||||
|
"""Creates a new job."""
|
||||||
|
params = job.to_dict()
|
||||||
|
if not POST_REQUIRED.issubset(set(params.keys())):
|
||||||
|
raise exc.InputException(
|
||||||
|
'Required param is missing. Required: %s' % POST_REQUIRED
|
||||||
|
)
|
||||||
|
|
||||||
|
first_time = params.get('first_execution_time')
|
||||||
|
pattern = params.get('pattern')
|
||||||
|
count = params.get('count')
|
||||||
|
start_time = timeutils.utcnow()
|
||||||
|
|
||||||
|
if isinstance(first_time, six.string_types):
|
||||||
|
try:
|
||||||
|
# We need naive datetime object.
|
||||||
|
first_time = parser.parse(first_time, ignoretz=True)
|
||||||
|
except ValueError as e:
|
||||||
|
raise exc.InputException(e.message)
|
||||||
|
|
||||||
|
if not (first_time or pattern):
|
||||||
|
raise exc.InputException(
|
||||||
|
'Pattern or first_execution_time must be specified.'
|
||||||
|
)
|
||||||
|
|
||||||
|
if first_time:
|
||||||
|
# first_time is assumed to be UTC time.
|
||||||
|
valid_min_time = timeutils.utcnow() + datetime.timedelta(0, 60)
|
||||||
|
if valid_min_time > first_time:
|
||||||
|
raise exc.InputException(
|
||||||
|
'first_execution_time must be at least 1 minute in the '
|
||||||
|
'future.'
|
||||||
|
)
|
||||||
|
if not pattern and count and count > 1:
|
||||||
|
raise exc.InputException(
|
||||||
|
'Pattern must be provided if count is greater than 1.'
|
||||||
|
)
|
||||||
|
|
||||||
|
next_time = first_time
|
||||||
|
if not (pattern or count):
|
||||||
|
count = 1
|
||||||
|
if pattern:
|
||||||
|
try:
|
||||||
|
croniter.croniter(pattern)
|
||||||
|
except (ValueError, KeyError):
|
||||||
|
raise exc.InputException(
|
||||||
|
'The specified pattern is not valid: {}'.format(pattern)
|
||||||
|
)
|
||||||
|
|
||||||
|
if not first_time:
|
||||||
|
next_time = croniter.croniter(pattern, start_time).get_next(
|
||||||
|
datetime.datetime
|
||||||
|
)
|
||||||
|
|
||||||
|
LOG.info("Creating job. [job=%s]", params)
|
||||||
|
|
||||||
|
with db_api.transaction():
|
||||||
|
db_api.get_function(params['function_id'])
|
||||||
|
|
||||||
|
values = {
|
||||||
|
'name': params.get('name'),
|
||||||
|
'pattern': pattern,
|
||||||
|
'first_execution_time': first_time,
|
||||||
|
'next_execution_time': next_time,
|
||||||
|
'count': count,
|
||||||
|
'function_id': params['function_id'],
|
||||||
|
'function_input': params.get('function_input') or {}
|
||||||
|
}
|
||||||
|
|
||||||
|
if cfg.CONF.pecan.auth_enable:
|
||||||
|
values['trust_id'] = keystone_util.create_trust().id
|
||||||
|
|
||||||
|
try:
|
||||||
|
db_job = db_api.create_job(values)
|
||||||
|
except Exception:
|
||||||
|
# Delete trust before raising exception.
|
||||||
|
keystone_util.delete_trust(values.get('trust_id'))
|
||||||
|
raise
|
||||||
|
|
||||||
|
return resources.Job.from_dict(db_job.to_dict())
|
||||||
|
|
||||||
|
@rest_utils.wrap_wsme_controller_exception
|
||||||
|
@wsme_pecan.wsexpose(None, types.uuid, status_code=204)
|
||||||
|
def delete(self, id):
|
||||||
|
"""Delete job."""
|
||||||
|
LOG.info("Delete job [id=%s]" % id)
|
||||||
|
|
||||||
|
job = db_api.get_job(id)
|
||||||
|
trust_id = job.trust_id
|
||||||
|
|
||||||
|
keystone_util.delete_trust(trust_id)
|
||||||
|
db_api.delete_job(id)
|
@ -170,6 +170,7 @@ class Function(Resource):
|
|||||||
code = types.jsontype
|
code = types.jsontype
|
||||||
entry = wtypes.text
|
entry = wtypes.text
|
||||||
count = wsme.wsattr(int, readonly=True)
|
count = wsme.wsattr(int, readonly=True)
|
||||||
|
project_id = wsme.wsattr(wtypes.text, readonly=True)
|
||||||
created_at = wtypes.text
|
created_at = wtypes.text
|
||||||
updated_at = wtypes.text
|
updated_at = wtypes.text
|
||||||
|
|
||||||
@ -185,6 +186,7 @@ class Function(Resource):
|
|||||||
code={'zip': True},
|
code={'zip': True},
|
||||||
entry='main',
|
entry='main',
|
||||||
count=10,
|
count=10,
|
||||||
|
project_id='default',
|
||||||
created_at='1970-01-01T00:00:00.000000',
|
created_at='1970-01-01T00:00:00.000000',
|
||||||
updated_at='1970-01-01T00:00:00.000000'
|
updated_at='1970-01-01T00:00:00.000000'
|
||||||
)
|
)
|
||||||
@ -263,6 +265,7 @@ class Execution(Resource):
|
|||||||
sync = bool
|
sync = bool
|
||||||
input = types.jsontype
|
input = types.jsontype
|
||||||
output = wsme.wsattr(types.jsontype, readonly=True)
|
output = wsme.wsattr(types.jsontype, readonly=True)
|
||||||
|
project_id = wsme.wsattr(wtypes.text, readonly=True)
|
||||||
created_at = wsme.wsattr(wtypes.text, readonly=True)
|
created_at = wsme.wsattr(wtypes.text, readonly=True)
|
||||||
updated_at = wsme.wsattr(wtypes.text, readonly=True)
|
updated_at = wsme.wsattr(wtypes.text, readonly=True)
|
||||||
|
|
||||||
@ -275,6 +278,7 @@ class Execution(Resource):
|
|||||||
sync=True,
|
sync=True,
|
||||||
input={'data': 'hello, world'},
|
input={'data': 'hello, world'},
|
||||||
output={'result': 'hello, world'},
|
output={'result': 'hello, world'},
|
||||||
|
project_id='default',
|
||||||
created_at='1970-01-01T00:00:00.000000',
|
created_at='1970-01-01T00:00:00.000000',
|
||||||
updated_at='1970-01-01T00:00:00.000000'
|
updated_at='1970-01-01T00:00:00.000000'
|
||||||
)
|
)
|
||||||
@ -299,3 +303,54 @@ class Executions(ResourceList):
|
|||||||
)
|
)
|
||||||
|
|
||||||
return sample
|
return sample
|
||||||
|
|
||||||
|
|
||||||
|
class Job(Resource):
|
||||||
|
id = types.uuid
|
||||||
|
name = wtypes.text
|
||||||
|
function_id = types.uuid
|
||||||
|
function_input = types.jsontype
|
||||||
|
pattern = wtypes.text
|
||||||
|
count = wtypes.IntegerType(minimum=1)
|
||||||
|
first_execution_time = wtypes.text
|
||||||
|
next_execution_time = wtypes.text
|
||||||
|
project_id = wsme.wsattr(wtypes.text, readonly=True)
|
||||||
|
created_at = wsme.wsattr(wtypes.text, readonly=True)
|
||||||
|
updated_at = wsme.wsattr(wtypes.text, readonly=True)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def sample(cls):
|
||||||
|
return cls(
|
||||||
|
id='123e4567-e89b-12d3-a456-426655440000',
|
||||||
|
name='my_job',
|
||||||
|
function_id='123e4567-e89b-12d3-a456-426655440000',
|
||||||
|
function_input={'data': 'hello, world'},
|
||||||
|
pattern='* * * * *',
|
||||||
|
count=42,
|
||||||
|
first_execution_time='',
|
||||||
|
next_execution_time='',
|
||||||
|
project_id='default',
|
||||||
|
created_at='1970-01-01T00:00:00.000000',
|
||||||
|
updated_at='1970-01-01T00:00:00.000000'
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class Jobs(ResourceList):
|
||||||
|
jobs = [Job]
|
||||||
|
|
||||||
|
def __init__(self, **kwargs):
|
||||||
|
self._type = 'jobs'
|
||||||
|
|
||||||
|
super(Jobs, self).__init__(**kwargs)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def sample(cls):
|
||||||
|
sample = cls()
|
||||||
|
sample.jobs = [Job.sample()]
|
||||||
|
sample.next = (
|
||||||
|
"http://localhost:7070/v1/jobs?"
|
||||||
|
"sort_keys=id,name&sort_dirs=asc,desc&limit=10&"
|
||||||
|
"marker=123e4567-e89b-12d3-a456-426655440000"
|
||||||
|
)
|
||||||
|
|
||||||
|
return sample
|
||||||
|
@ -18,6 +18,7 @@ import wsmeext.pecan as wsme_pecan
|
|||||||
|
|
||||||
from qinling.api.controllers.v1 import execution
|
from qinling.api.controllers.v1 import execution
|
||||||
from qinling.api.controllers.v1 import function
|
from qinling.api.controllers.v1 import function
|
||||||
|
from qinling.api.controllers.v1 import job
|
||||||
from qinling.api.controllers.v1 import resources
|
from qinling.api.controllers.v1 import resources
|
||||||
from qinling.api.controllers.v1 import runtime
|
from qinling.api.controllers.v1 import runtime
|
||||||
|
|
||||||
@ -37,6 +38,7 @@ class Controller(object):
|
|||||||
functions = function.FunctionsController()
|
functions = function.FunctionsController()
|
||||||
runtimes = runtime.RuntimesController()
|
runtimes = runtime.RuntimesController()
|
||||||
executions = execution.ExecutionsController()
|
executions = execution.ExecutionsController()
|
||||||
|
jobs = job.JobsController()
|
||||||
|
|
||||||
@wsme_pecan.wsexpose(RootResource)
|
@wsme_pecan.wsexpose(RootResource)
|
||||||
def index(self):
|
def index(self):
|
||||||
|
@ -159,7 +159,11 @@ _DEFAULT_LOG_LEVELS = [
|
|||||||
'oslo_service.periodic_task=INFO',
|
'oslo_service.periodic_task=INFO',
|
||||||
'oslo_service.loopingcall=INFO',
|
'oslo_service.loopingcall=INFO',
|
||||||
'oslo_db=WARN',
|
'oslo_db=WARN',
|
||||||
'oslo_concurrency.lockutils=WARN'
|
'oslo_concurrency.lockutils=WARN',
|
||||||
|
'kubernetes.client.rest=DEBUG',
|
||||||
|
'keystoneclient=INFO',
|
||||||
|
'requests.packages.urllib3.connectionpool=CRITICAL',
|
||||||
|
'urllib3.connectionpool=CRITICAL'
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
|
@ -59,9 +59,6 @@ def delete_all():
|
|||||||
delete_runtimes(insecure=True)
|
delete_runtimes(insecure=True)
|
||||||
|
|
||||||
|
|
||||||
# Function
|
|
||||||
|
|
||||||
|
|
||||||
def get_function(id):
|
def get_function(id):
|
||||||
return IMPL.get_function(id)
|
return IMPL.get_function(id)
|
||||||
|
|
||||||
@ -90,9 +87,6 @@ def delete_function(id):
|
|||||||
IMPL.delete_function(id)
|
IMPL.delete_function(id)
|
||||||
|
|
||||||
|
|
||||||
# Function
|
|
||||||
|
|
||||||
|
|
||||||
def create_runtime(values):
|
def create_runtime(values):
|
||||||
return IMPL.create_runtime(values)
|
return IMPL.create_runtime(values)
|
||||||
|
|
||||||
@ -117,9 +111,6 @@ def delete_runtimes(**kwargs):
|
|||||||
return IMPL.delete_runtimes(**kwargs)
|
return IMPL.delete_runtimes(**kwargs)
|
||||||
|
|
||||||
|
|
||||||
# Execution
|
|
||||||
|
|
||||||
|
|
||||||
def create_execution(values):
|
def create_execution(values):
|
||||||
return IMPL.create_execution(values)
|
return IMPL.create_execution(values)
|
||||||
|
|
||||||
@ -154,3 +145,15 @@ def get_function_service_mappings(**kwargs):
|
|||||||
|
|
||||||
def delete_function_service_mapping(id):
|
def delete_function_service_mapping(id):
|
||||||
return IMPL.delete_function_service_mapping(id)
|
return IMPL.delete_function_service_mapping(id)
|
||||||
|
|
||||||
|
|
||||||
|
def create_job(values):
|
||||||
|
return IMPL.create_job(values)
|
||||||
|
|
||||||
|
|
||||||
|
def get_job(id):
|
||||||
|
return IMPL.get_job(id)
|
||||||
|
|
||||||
|
|
||||||
|
def delete_job(id):
|
||||||
|
return IMPL.delete_job(id)
|
||||||
|
@ -366,3 +366,34 @@ def delete_function_service_mapping(id, session=None):
|
|||||||
return
|
return
|
||||||
|
|
||||||
session.delete(mapping)
|
session.delete(mapping)
|
||||||
|
|
||||||
|
|
||||||
|
@db_base.session_aware()
|
||||||
|
def create_job(values, session=None):
|
||||||
|
job = models.Job()
|
||||||
|
job.update(values)
|
||||||
|
|
||||||
|
try:
|
||||||
|
job.save(session=session)
|
||||||
|
except oslo_db_exc.DBDuplicateEntry as e:
|
||||||
|
raise exc.DBError(
|
||||||
|
"Duplicate entry for Job: %s" % e.columns
|
||||||
|
)
|
||||||
|
|
||||||
|
return job
|
||||||
|
|
||||||
|
|
||||||
|
@db_base.session_aware()
|
||||||
|
def get_job(id, session=None):
|
||||||
|
job = _get_db_object_by_id(models.Job, id)
|
||||||
|
if not job:
|
||||||
|
raise exc.DBEntityNotFoundError("Job not found [id=%s]" % id)
|
||||||
|
|
||||||
|
return job
|
||||||
|
|
||||||
|
|
||||||
|
@db_base.session_aware()
|
||||||
|
def delete_job(id, session=None):
|
||||||
|
job = get_job(id)
|
||||||
|
|
||||||
|
session.delete(job)
|
||||||
|
@ -107,3 +107,22 @@ def upgrade():
|
|||||||
sa.PrimaryKeyConstraint('id'),
|
sa.PrimaryKeyConstraint('id'),
|
||||||
info={"check_ifexists": True}
|
info={"check_ifexists": True}
|
||||||
)
|
)
|
||||||
|
|
||||||
|
op.create_table(
|
||||||
|
'job',
|
||||||
|
sa.Column('created_at', sa.DateTime(), nullable=True),
|
||||||
|
sa.Column('updated_at', sa.DateTime(), nullable=True),
|
||||||
|
sa.Column('project_id', sa.String(length=80), nullable=False),
|
||||||
|
sa.Column('id', sa.String(length=36), nullable=False),
|
||||||
|
sa.Column('function_id', sa.String(length=36), nullable=False),
|
||||||
|
sa.Column('function_input', st.JsonLongDictType(), nullable=True),
|
||||||
|
sa.Column('name', sa.String(length=255), nullable=True),
|
||||||
|
sa.Column('pattern', sa.String(length=32), nullable=False),
|
||||||
|
sa.Column('first_execution_time', sa.DateTime(), nullable=True),
|
||||||
|
sa.Column('next_execution_time', sa.DateTime(), nullable=False),
|
||||||
|
sa.Column('count', sa.Integer(), nullable=True),
|
||||||
|
sa.Column('trust_id', sa.String(length=80), nullable=True),
|
||||||
|
sa.PrimaryKeyConstraint('id'),
|
||||||
|
sa.ForeignKeyConstraint(['function_id'], [u'function.id']),
|
||||||
|
info={"check_ifexists": True}
|
||||||
|
)
|
||||||
|
@ -12,8 +12,6 @@
|
|||||||
# See the License for the specific language governing permissions and
|
# See the License for the specific language governing permissions and
|
||||||
# limitations under the License.
|
# limitations under the License.
|
||||||
|
|
||||||
import six
|
|
||||||
|
|
||||||
from oslo_db.sqlalchemy import models as oslo_models
|
from oslo_db.sqlalchemy import models as oslo_models
|
||||||
import sqlalchemy as sa
|
import sqlalchemy as sa
|
||||||
from sqlalchemy.ext import declarative
|
from sqlalchemy.ext import declarative
|
||||||
@ -76,8 +74,8 @@ class _QinlingModelBase(oslo_models.ModelBase, oslo_models.TimestampMixin):
|
|||||||
if col.name not in unloaded and hasattr(self, col.name):
|
if col.name not in unloaded and hasattr(self, col.name):
|
||||||
d[col.name] = getattr(self, col.name)
|
d[col.name] = getattr(self, col.name)
|
||||||
|
|
||||||
datetime_to_str(d, 'created_at')
|
common.datetime_to_str(d, 'created_at')
|
||||||
datetime_to_str(d, 'updated_at')
|
common.datetime_to_str(d, 'updated_at')
|
||||||
|
|
||||||
return d
|
return d
|
||||||
|
|
||||||
@ -101,12 +99,6 @@ class _QinlingModelBase(oslo_models.ModelBase, oslo_models.TimestampMixin):
|
|||||||
return '%s %s' % (type(self).__name__, self.to_dict().__repr__())
|
return '%s %s' % (type(self).__name__, self.to_dict().__repr__())
|
||||||
|
|
||||||
|
|
||||||
def datetime_to_str(dct, attr_name):
|
|
||||||
if (dct.get(attr_name) is not None and
|
|
||||||
not isinstance(dct.get(attr_name), six.string_types)):
|
|
||||||
dct[attr_name] = dct[attr_name].isoformat(' ')
|
|
||||||
|
|
||||||
|
|
||||||
QinlingModelBase = declarative.declarative_base(cls=_QinlingModelBase)
|
QinlingModelBase = declarative.declarative_base(cls=_QinlingModelBase)
|
||||||
|
|
||||||
|
|
||||||
|
@ -13,9 +13,11 @@
|
|||||||
# limitations under the License.
|
# limitations under the License.
|
||||||
|
|
||||||
import sqlalchemy as sa
|
import sqlalchemy as sa
|
||||||
|
from sqlalchemy.orm import relationship
|
||||||
|
|
||||||
from qinling.db.sqlalchemy import model_base
|
from qinling.db.sqlalchemy import model_base
|
||||||
from qinling.db.sqlalchemy import types as st
|
from qinling.db.sqlalchemy import types as st
|
||||||
|
from qinling.utils import common
|
||||||
|
|
||||||
|
|
||||||
class Function(model_base.QinlingSecureModelBase):
|
class Function(model_base.QinlingSecureModelBase):
|
||||||
@ -71,3 +73,31 @@ class Execution(model_base.QinlingSecureModelBase):
|
|||||||
sync = sa.Column(sa.BOOLEAN, default=True)
|
sync = sa.Column(sa.BOOLEAN, default=True)
|
||||||
input = sa.Column(st.JsonLongDictType())
|
input = sa.Column(st.JsonLongDictType())
|
||||||
output = sa.Column(st.JsonLongDictType())
|
output = sa.Column(st.JsonLongDictType())
|
||||||
|
|
||||||
|
|
||||||
|
class Job(model_base.QinlingSecureModelBase):
|
||||||
|
__tablename__ = 'job'
|
||||||
|
|
||||||
|
name = sa.Column(sa.String(255), nullable=True)
|
||||||
|
pattern = sa.Column(
|
||||||
|
sa.String(32),
|
||||||
|
nullable=True,
|
||||||
|
# Set default to 'never'.
|
||||||
|
default='0 0 30 2 0'
|
||||||
|
)
|
||||||
|
first_execution_time = sa.Column(sa.DateTime, nullable=True)
|
||||||
|
next_execution_time = sa.Column(sa.DateTime, nullable=False)
|
||||||
|
count = sa.Column(sa.Integer)
|
||||||
|
function_id = sa.Column(
|
||||||
|
sa.String(36),
|
||||||
|
sa.ForeignKey(Function.id)
|
||||||
|
)
|
||||||
|
function = relationship('Function', lazy='joined')
|
||||||
|
function_input = sa.Column(st.JsonDictType())
|
||||||
|
trust_id = sa.Column(sa.String(80))
|
||||||
|
|
||||||
|
def to_dict(self):
|
||||||
|
d = super(Job, self).to_dict()
|
||||||
|
common.datetime_to_str(d, 'first_execution_time')
|
||||||
|
common.datetime_to_str(d, 'next_execution_time')
|
||||||
|
return d
|
||||||
|
48
qinling/tests/unit/api/controllers/v1/test_job.py
Normal file
48
qinling/tests/unit/api/controllers/v1/test_job.py
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
# Copyright 2017 Catalyst IT Limited
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
# you may not use this file except in compliance with the License.
|
||||||
|
# You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
|
||||||
|
from datetime import datetime
|
||||||
|
from datetime import timedelta
|
||||||
|
|
||||||
|
from qinling.tests.unit.api import base
|
||||||
|
|
||||||
|
|
||||||
|
class TestJobController(base.APITest):
|
||||||
|
def setUp(self):
|
||||||
|
super(TestJobController, self).setUp()
|
||||||
|
|
||||||
|
# Insert a function record in db for each test case. The data will be
|
||||||
|
# removed automatically in db clean up.
|
||||||
|
db_function = self.create_function(prefix='TestJobController')
|
||||||
|
self.function_id = db_function.id
|
||||||
|
|
||||||
|
def test_post(self):
|
||||||
|
body = {
|
||||||
|
'name': self.rand_name('job', prefix='TestJobController'),
|
||||||
|
'first_execution_time': str(
|
||||||
|
datetime.utcnow() + timedelta(hours=1)),
|
||||||
|
'function_id': self.function_id
|
||||||
|
}
|
||||||
|
resp = self.app.post_json('/v1/jobs', body)
|
||||||
|
|
||||||
|
self.assertEqual(201, resp.status_int)
|
||||||
|
|
||||||
|
def test_delete(self):
|
||||||
|
job_id = self.create_job(
|
||||||
|
self.function_id, prefix='TestJobController'
|
||||||
|
).id
|
||||||
|
|
||||||
|
resp = self.app.delete('/v1/jobs/%s' % job_id)
|
||||||
|
|
||||||
|
self.assertEqual(204, resp.status_int)
|
@ -26,16 +26,7 @@ class TestRuntimeController(base.APITest):
|
|||||||
|
|
||||||
# Insert a runtime record in db. The data will be removed in db clean
|
# Insert a runtime record in db. The data will be removed in db clean
|
||||||
# up.
|
# up.
|
||||||
self.db_runtime = db_api.create_runtime(
|
self.db_runtime = self.create_runtime(prefix='TestRuntimeController')
|
||||||
{
|
|
||||||
'name': 'test_runtime',
|
|
||||||
'image': 'python2.7',
|
|
||||||
# 'auth_enable' is disabled by default, we create runtime for
|
|
||||||
# default tenant.
|
|
||||||
'project_id': test_base.DEFAULT_PROJECT_ID,
|
|
||||||
'status': status.AVAILABLE
|
|
||||||
}
|
|
||||||
)
|
|
||||||
self.runtime_id = self.db_runtime.id
|
self.runtime_id = self.db_runtime.id
|
||||||
|
|
||||||
def test_get(self):
|
def test_get(self):
|
||||||
@ -43,8 +34,8 @@ class TestRuntimeController(base.APITest):
|
|||||||
|
|
||||||
expected = {
|
expected = {
|
||||||
'id': self.runtime_id,
|
'id': self.runtime_id,
|
||||||
"image": "python2.7",
|
"image": self.db_runtime.image,
|
||||||
"name": "test_runtime",
|
"name": self.db_runtime.name,
|
||||||
"project_id": test_base.DEFAULT_PROJECT_ID,
|
"project_id": test_base.DEFAULT_PROJECT_ID,
|
||||||
"status": status.AVAILABLE
|
"status": status.AVAILABLE
|
||||||
}
|
}
|
||||||
@ -57,8 +48,8 @@ class TestRuntimeController(base.APITest):
|
|||||||
|
|
||||||
expected = {
|
expected = {
|
||||||
'id': self.runtime_id,
|
'id': self.runtime_id,
|
||||||
"image": "python2.7",
|
"image": self.db_runtime.image,
|
||||||
"name": "test_runtime",
|
"name": self.db_runtime.name,
|
||||||
"project_id": test_base.DEFAULT_PROJECT_ID,
|
"project_id": test_base.DEFAULT_PROJECT_ID,
|
||||||
"status": status.AVAILABLE
|
"status": status.AVAILABLE
|
||||||
}
|
}
|
||||||
@ -72,8 +63,8 @@ class TestRuntimeController(base.APITest):
|
|||||||
@mock.patch('qinling.rpc.EngineClient.create_runtime')
|
@mock.patch('qinling.rpc.EngineClient.create_runtime')
|
||||||
def test_post(self, mock_create_time):
|
def test_post(self, mock_create_time):
|
||||||
body = {
|
body = {
|
||||||
'name': self.rand_name('runtime', prefix='APITest'),
|
'name': self.rand_name('runtime', prefix='TestRuntimeController'),
|
||||||
'image': self.rand_name('image', prefix='APITest'),
|
'image': self.rand_name('image', prefix='TestRuntimeController'),
|
||||||
}
|
}
|
||||||
resp = self.app.post_json('/v1/runtimes', body)
|
resp = self.app.post_json('/v1/runtimes', body)
|
||||||
|
|
||||||
@ -83,36 +74,13 @@ class TestRuntimeController(base.APITest):
|
|||||||
|
|
||||||
@mock.patch('qinling.rpc.EngineClient.delete_runtime')
|
@mock.patch('qinling.rpc.EngineClient.delete_runtime')
|
||||||
def test_delete(self, mock_delete_runtime):
|
def test_delete(self, mock_delete_runtime):
|
||||||
db_runtime = db_api.create_runtime(
|
resp = self.app.delete('/v1/runtimes/%s' % self.runtime_id)
|
||||||
{
|
|
||||||
'name': self.rand_name('runtime', prefix='APITest'),
|
|
||||||
'image': self.rand_name('image', prefix='APITest'),
|
|
||||||
# 'auth_enable' is disabled by default, we create runtime for
|
|
||||||
# default tenant.
|
|
||||||
'project_id': test_base.DEFAULT_PROJECT_ID,
|
|
||||||
'status': status.AVAILABLE
|
|
||||||
}
|
|
||||||
)
|
|
||||||
runtime_id = db_runtime.id
|
|
||||||
|
|
||||||
resp = self.app.delete('/v1/runtimes/%s' % runtime_id)
|
|
||||||
|
|
||||||
self.assertEqual(204, resp.status_int)
|
self.assertEqual(204, resp.status_int)
|
||||||
mock_delete_runtime.assert_called_once_with(runtime_id)
|
mock_delete_runtime.assert_called_once_with(self.runtime_id)
|
||||||
|
|
||||||
def test_delete_runtime_with_function_associated(self):
|
def test_delete_runtime_with_function_associated(self):
|
||||||
db_api.create_function(
|
self.create_function(self.runtime_id, prefix='TestRuntimeController')
|
||||||
{
|
|
||||||
'name': self.rand_name('function', prefix='APITest'),
|
|
||||||
'runtime_id': self.runtime_id,
|
|
||||||
'code': {},
|
|
||||||
'entry': 'main.main',
|
|
||||||
# 'auth_enable' is disabled by default, we create runtime for
|
|
||||||
# default tenant.
|
|
||||||
'project_id': test_base.DEFAULT_PROJECT_ID,
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
resp = self.app.delete(
|
resp = self.app.delete(
|
||||||
'/v1/runtimes/%s' % self.runtime_id, expect_errors=True
|
'/v1/runtimes/%s' % self.runtime_id, expect_errors=True
|
||||||
)
|
)
|
||||||
@ -130,8 +98,10 @@ class TestRuntimeController(base.APITest):
|
|||||||
def test_put_image_runtime_not_available(self):
|
def test_put_image_runtime_not_available(self):
|
||||||
db_runtime = db_api.create_runtime(
|
db_runtime = db_api.create_runtime(
|
||||||
{
|
{
|
||||||
'name': self.rand_name('runtime', prefix='APITest'),
|
'name': self.rand_name(
|
||||||
'image': self.rand_name('image', prefix='APITest'),
|
'runtime', prefix='TestRuntimeController'),
|
||||||
|
'image': self.rand_name(
|
||||||
|
'image', prefix='TestRuntimeController'),
|
||||||
'project_id': test_base.DEFAULT_PROJECT_ID,
|
'project_id': test_base.DEFAULT_PROJECT_ID,
|
||||||
'status': status.CREATING
|
'status': status.CREATING
|
||||||
}
|
}
|
||||||
|
@ -14,6 +14,7 @@
|
|||||||
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
# License for the specific language governing permissions and limitations
|
# License for the specific language governing permissions and limitations
|
||||||
# under the License.
|
# under the License.
|
||||||
|
from datetime import datetime
|
||||||
import random
|
import random
|
||||||
|
|
||||||
from oslo_config import cfg
|
from oslo_config import cfg
|
||||||
@ -22,6 +23,7 @@ from oslotest import base
|
|||||||
from qinling import context as auth_context
|
from qinling import context as auth_context
|
||||||
from qinling.db import api as db_api
|
from qinling.db import api as db_api
|
||||||
from qinling.db.sqlalchemy import sqlite_lock
|
from qinling.db.sqlalchemy import sqlite_lock
|
||||||
|
from qinling import status
|
||||||
from qinling.tests.unit import config as test_config
|
from qinling.tests.unit import config as test_config
|
||||||
|
|
||||||
test_config.parse_args()
|
test_config.parse_args()
|
||||||
@ -151,3 +153,54 @@ class DbTestCase(BaseTest):
|
|||||||
def _clean_db(self):
|
def _clean_db(self):
|
||||||
db_api.delete_all()
|
db_api.delete_all()
|
||||||
sqlite_lock.cleanup()
|
sqlite_lock.cleanup()
|
||||||
|
|
||||||
|
def create_runtime(self, prefix=None):
|
||||||
|
runtime = db_api.create_runtime(
|
||||||
|
{
|
||||||
|
'name': self.rand_name('runtime', prefix=prefix),
|
||||||
|
'image': self.rand_name('image', prefix=prefix),
|
||||||
|
# 'auth_enable' is disabled by default, we create runtime for
|
||||||
|
# default tenant.
|
||||||
|
'project_id': DEFAULT_PROJECT_ID,
|
||||||
|
'status': status.AVAILABLE
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
return runtime
|
||||||
|
|
||||||
|
def create_function(self, runtime_id=None, prefix=None):
|
||||||
|
if not runtime_id:
|
||||||
|
runtime_id = self.create_runtime(prefix).id
|
||||||
|
|
||||||
|
function = db_api.create_function(
|
||||||
|
{
|
||||||
|
'name': self.rand_name('function', prefix=prefix),
|
||||||
|
'runtime_id': runtime_id,
|
||||||
|
'code': {"source": "package"},
|
||||||
|
'entry': 'main.main',
|
||||||
|
# 'auth_enable' is disabled by default, we create runtime for
|
||||||
|
# default tenant.
|
||||||
|
'project_id': DEFAULT_PROJECT_ID,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
return function
|
||||||
|
|
||||||
|
def create_job(self, function_id=None, prefix=None):
|
||||||
|
if not function_id:
|
||||||
|
function_id = self.create_function(prefix=prefix).id
|
||||||
|
|
||||||
|
job = db_api.create_job(
|
||||||
|
{
|
||||||
|
'name': self.rand_name('job', prefix=prefix),
|
||||||
|
'function_id': function_id,
|
||||||
|
'first_execution_time': datetime.utcnow(),
|
||||||
|
'next_execution_time': datetime.utcnow(),
|
||||||
|
'count': 1,
|
||||||
|
# 'auth_enable' is disabled by default, we create runtime for
|
||||||
|
# default tenant.
|
||||||
|
'project_id': DEFAULT_PROJECT_ID,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
return job
|
||||||
|
@ -15,6 +15,7 @@ import functools
|
|||||||
import warnings
|
import warnings
|
||||||
|
|
||||||
from oslo_utils import uuidutils
|
from oslo_utils import uuidutils
|
||||||
|
import six
|
||||||
|
|
||||||
|
|
||||||
def convert_dict_to_string(d):
|
def convert_dict_to_string(d):
|
||||||
@ -23,6 +24,13 @@ def convert_dict_to_string(d):
|
|||||||
return ','.join(temp_list)
|
return ','.join(temp_list)
|
||||||
|
|
||||||
|
|
||||||
|
def datetime_to_str(dct, attr_name):
|
||||||
|
"""Convert datetime object in dict to string."""
|
||||||
|
if (dct.get(attr_name) is not None and
|
||||||
|
not isinstance(dct.get(attr_name), six.string_types)):
|
||||||
|
dct[attr_name] = dct[attr_name].isoformat(' ')
|
||||||
|
|
||||||
|
|
||||||
def generate_unicode_uuid(dashed=True):
|
def generate_unicode_uuid(dashed=True):
|
||||||
return uuidutils.generate_uuid(dashed=dashed)
|
return uuidutils.generate_uuid(dashed=dashed)
|
||||||
|
|
||||||
|
@ -14,11 +14,15 @@
|
|||||||
|
|
||||||
from keystoneauth1.identity import generic
|
from keystoneauth1.identity import generic
|
||||||
from keystoneauth1 import session
|
from keystoneauth1 import session
|
||||||
|
from keystoneclient.v3 import client as ks_client
|
||||||
from oslo_config import cfg
|
from oslo_config import cfg
|
||||||
|
from oslo_log import log as logging
|
||||||
import swiftclient
|
import swiftclient
|
||||||
|
|
||||||
from qinling import context
|
from qinling import context
|
||||||
|
from qinling.utils import common
|
||||||
|
|
||||||
|
LOG = logging.getLogger(__name__)
|
||||||
CONF = cfg.CONF
|
CONF = cfg.CONF
|
||||||
|
|
||||||
|
|
||||||
@ -33,9 +37,58 @@ def _get_user_keystone_session():
|
|||||||
return session.Session(auth=auth, verify=False)
|
return session.Session(auth=auth, verify=False)
|
||||||
|
|
||||||
|
|
||||||
|
@common.disable_ssl_warnings
|
||||||
def get_swiftclient():
|
def get_swiftclient():
|
||||||
session = _get_user_keystone_session()
|
session = _get_user_keystone_session()
|
||||||
|
|
||||||
conn = swiftclient.Connection(session=session)
|
conn = swiftclient.Connection(session=session)
|
||||||
|
|
||||||
return conn
|
return conn
|
||||||
|
|
||||||
|
|
||||||
|
@common.disable_ssl_warnings
|
||||||
|
def get_keystone_client():
|
||||||
|
session = _get_user_keystone_session()
|
||||||
|
keystone = ks_client.Client(session=session)
|
||||||
|
|
||||||
|
return keystone
|
||||||
|
|
||||||
|
|
||||||
|
@common.disable_ssl_warnings
|
||||||
|
def _get_admin_user_id():
|
||||||
|
auth_url = CONF.keystone_authtoken.auth_uri
|
||||||
|
client = ks_client.Client(
|
||||||
|
username=CONF.keystone_authtoken.username,
|
||||||
|
password=CONF.keystone_authtoken.password,
|
||||||
|
project_name=CONF.keystone_authtoken.project_name,
|
||||||
|
auth_url=auth_url,
|
||||||
|
)
|
||||||
|
|
||||||
|
return client.user_id
|
||||||
|
|
||||||
|
|
||||||
|
@common.disable_ssl_warnings
|
||||||
|
def create_trust():
|
||||||
|
client = get_keystone_client()
|
||||||
|
ctx = context.get_ctx()
|
||||||
|
trustee_id = _get_admin_user_id()
|
||||||
|
|
||||||
|
return client.trusts.create(
|
||||||
|
trustor_user=ctx.user,
|
||||||
|
trustee_user=trustee_id,
|
||||||
|
impersonation=True,
|
||||||
|
role_names=ctx.roles,
|
||||||
|
project=ctx.tenant
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@common.disable_ssl_warnings
|
||||||
|
def delete_trust(trust_id):
|
||||||
|
if not trust_id:
|
||||||
|
return
|
||||||
|
|
||||||
|
client = get_keystone_client()
|
||||||
|
try:
|
||||||
|
client.trusts.delete(trust_id)
|
||||||
|
except Exception as e:
|
||||||
|
LOG.warning("Failed to delete trust [id=%s]: %s" % (trust_id, e))
|
||||||
|
@ -26,3 +26,5 @@ WSME>=0.8 # MIT
|
|||||||
kubernetes>=1.0.0b1 # Apache-2.0
|
kubernetes>=1.0.0b1 # Apache-2.0
|
||||||
PyYAML>=3.10.0 # MIT
|
PyYAML>=3.10.0 # MIT
|
||||||
python-swiftclient>=3.2.0 # Apache-2.0
|
python-swiftclient>=3.2.0 # Apache-2.0
|
||||||
|
croniter>=0.3.4 # MIT License
|
||||||
|
python-dateutil>=2.4.2 # BSD
|
||||||
|
Loading…
Reference in New Issue
Block a user