Merge "Fix the TTL issue of subscriptions for MongoDB"
This commit is contained in:
commit
883bade05a
@ -58,7 +58,7 @@ class RequestSchema(v1_1.RequestSchema):
|
|||||||
'ttl': {'type': 'integer'},
|
'ttl': {'type': 'integer'},
|
||||||
'options': {'type': 'object'},
|
'options': {'type': 'object'},
|
||||||
},
|
},
|
||||||
'required': ['queue_name', 'ttl'],
|
'required': ['queue_name', ],
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
'required': ['action', 'headers', 'body']
|
'required': ['action', 'headers', 'body']
|
||||||
|
@ -11,6 +11,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 under
|
# License for the specific language governing permissions and limitations under
|
||||||
# the License.
|
# the License.
|
||||||
|
import datetime
|
||||||
|
|
||||||
from oslo_utils import timeutils
|
from oslo_utils import timeutils
|
||||||
import pymongo.errors
|
import pymongo.errors
|
||||||
@ -29,6 +30,11 @@ SUBSCRIPTIONS_INDEX = [
|
|||||||
('p', 1),
|
('p', 1),
|
||||||
]
|
]
|
||||||
|
|
||||||
|
# For removing expired subscriptions
|
||||||
|
TTL_INDEX_FIELDS = [
|
||||||
|
('e', 1),
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
class SubscriptionController(base.Subscription):
|
class SubscriptionController(base.Subscription):
|
||||||
"""Implements subscription resource operations using MongoDB.
|
"""Implements subscription resource operations using MongoDB.
|
||||||
@ -49,6 +55,14 @@ class SubscriptionController(base.Subscription):
|
|||||||
self._collection = self.driver.subscriptions_database.subscriptions
|
self._collection = self.driver.subscriptions_database.subscriptions
|
||||||
self._queue_ctrl = self.driver.queue_controller
|
self._queue_ctrl = self.driver.queue_controller
|
||||||
self._collection.ensure_index(SUBSCRIPTIONS_INDEX, unique=True)
|
self._collection.ensure_index(SUBSCRIPTIONS_INDEX, unique=True)
|
||||||
|
# NOTE(flwang): MongoDB will automatically delete the subscription
|
||||||
|
# from the subscriptions collection when the subscription's 'e' value
|
||||||
|
# is older than the number of seconds specified in expireAfterSeconds,
|
||||||
|
# i.e. 0 seconds older in this case. As such, the data expires at the
|
||||||
|
# specified 'e' value.
|
||||||
|
self._collection.ensure_index(TTL_INDEX_FIELDS, name='ttl',
|
||||||
|
expireAfterSeconds=0,
|
||||||
|
background=True)
|
||||||
|
|
||||||
@utils.raises_conn_error
|
@utils.raises_conn_error
|
||||||
def list(self, queue, project=None, marker=None,
|
def list(self, queue, project=None, marker=None,
|
||||||
@ -92,8 +106,8 @@ class SubscriptionController(base.Subscription):
|
|||||||
def create(self, queue, subscriber, ttl, options, project=None):
|
def create(self, queue, subscriber, ttl, options, project=None):
|
||||||
source = queue
|
source = queue
|
||||||
now = timeutils.utcnow_ts()
|
now = timeutils.utcnow_ts()
|
||||||
ttl = int(ttl)
|
now_dt = datetime.datetime.utcfromtimestamp(now)
|
||||||
expires = now + ttl
|
expires = now_dt + datetime.timedelta(seconds=ttl)
|
||||||
|
|
||||||
if not self._queue_ctrl.exists(source, project):
|
if not self._queue_ctrl.exists(source, project):
|
||||||
raise errors.QueueDoesNotExist(source, project)
|
raise errors.QueueDoesNotExist(source, project)
|
||||||
|
@ -97,4 +97,19 @@ class TestValidation(base.V2Base):
|
|||||||
self.project_id,
|
self.project_id,
|
||||||
body='{"timespace": "Shangri-la"}',
|
body='{"timespace": "Shangri-la"}',
|
||||||
headers=empty_headers)
|
headers=empty_headers)
|
||||||
|
|
||||||
|
def test_subscription_ttl(self):
|
||||||
|
# Normal case
|
||||||
|
body = '{"subscriber": "http://trigger.she", "ttl": 100, "options":{}}'
|
||||||
|
self.simulate_post(self.queue_path + '/subscriptions',
|
||||||
|
self.project_id, body=body,
|
||||||
|
headers=self.headers)
|
||||||
|
self.assertEqual(falcon.HTTP_201, self.srmock.status)
|
||||||
|
|
||||||
|
# Very big TTL
|
||||||
|
body = ('{"subscriber": "http://a.c", "ttl": 99999999999999999'
|
||||||
|
', "options":{}}')
|
||||||
|
self.simulate_post(self.queue_path + '/subscriptions',
|
||||||
|
self.project_id, body=body,
|
||||||
|
headers=self.headers)
|
||||||
self.assertEqual(falcon.HTTP_400, self.srmock.status)
|
self.assertEqual(falcon.HTTP_400, self.srmock.status)
|
||||||
|
@ -33,7 +33,9 @@ _RESOURCE_DEFAULTS = (
|
|||||||
cfg.IntOpt('default_claim_ttl', default=300,
|
cfg.IntOpt('default_claim_ttl', default=300,
|
||||||
help=('Defines how long a message will be in claimed state.')),
|
help=('Defines how long a message will be in claimed state.')),
|
||||||
cfg.IntOpt('default_claim_grace', default=60,
|
cfg.IntOpt('default_claim_grace', default=60,
|
||||||
help=('Defines the message grace period in seconds.'))
|
help=('Defines the message grace period in seconds.')),
|
||||||
|
cfg.IntOpt('default_subscription_ttl', default=3600,
|
||||||
|
help=('Defines how long a subscription will be available.')),
|
||||||
)
|
)
|
||||||
|
|
||||||
_TRANSPORT_GROUP = 'transport'
|
_TRANSPORT_GROUP = 'transport'
|
||||||
@ -66,6 +68,10 @@ class ResourceDefaults(object):
|
|||||||
def claim_grace(self):
|
def claim_grace(self):
|
||||||
return self._defaults.default_claim_grace
|
return self._defaults.default_claim_grace
|
||||||
|
|
||||||
|
@property
|
||||||
|
def subscription_ttl(self):
|
||||||
|
return self._defaults.default_subscription_ttl
|
||||||
|
|
||||||
|
|
||||||
@six.add_metaclass(abc.ABCMeta)
|
@six.add_metaclass(abc.ABCMeta)
|
||||||
class DriverBase(object):
|
class DriverBase(object):
|
||||||
|
@ -14,9 +14,11 @@
|
|||||||
# 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 datetime
|
||||||
import re
|
import re
|
||||||
|
|
||||||
from oslo_config import cfg
|
from oslo_config import cfg
|
||||||
|
from oslo_utils import timeutils
|
||||||
import six
|
import six
|
||||||
|
|
||||||
from zaqar.i18n import _
|
from zaqar.i18n import _
|
||||||
@ -24,6 +26,7 @@ from zaqar.i18n import _
|
|||||||
MIN_MESSAGE_TTL = 60
|
MIN_MESSAGE_TTL = 60
|
||||||
MIN_CLAIM_TTL = 60
|
MIN_CLAIM_TTL = 60
|
||||||
MIN_CLAIM_GRACE = 60
|
MIN_CLAIM_GRACE = 60
|
||||||
|
MIN_SUBSCRIPTION_TTL = 60
|
||||||
|
|
||||||
_TRANSPORT_LIMITS_OPTIONS = (
|
_TRANSPORT_LIMITS_OPTIONS = (
|
||||||
cfg.IntOpt('max_queues_per_page', default=20,
|
cfg.IntOpt('max_queues_per_page', default=20,
|
||||||
@ -295,7 +298,7 @@ class Validator(object):
|
|||||||
:param subscription: dict of subscription
|
:param subscription: dict of subscription
|
||||||
:raises: ValidationFailed if the subscription is invalid.
|
:raises: ValidationFailed if the subscription is invalid.
|
||||||
"""
|
"""
|
||||||
for p in ('subscriber', 'ttl', 'options'):
|
for p in ('subscriber',):
|
||||||
if p not in subscription.keys():
|
if p not in subscription.keys():
|
||||||
raise ValidationFailed(_(u'Missing parameter %s in body.') % p)
|
raise ValidationFailed(_(u'Missing parameter %s in body.') % p)
|
||||||
|
|
||||||
@ -329,10 +332,30 @@ class Validator(object):
|
|||||||
raise ValidationFailed(msg)
|
raise ValidationFailed(msg)
|
||||||
|
|
||||||
ttl = subscription.get('ttl', None)
|
ttl = subscription.get('ttl', None)
|
||||||
if ttl and not isinstance(ttl, int):
|
if ttl:
|
||||||
|
if not isinstance(ttl, int):
|
||||||
msg = _(u'TTL must be an integer.')
|
msg = _(u'TTL must be an integer.')
|
||||||
raise ValidationFailed(msg)
|
raise ValidationFailed(msg)
|
||||||
|
|
||||||
|
if ttl < MIN_SUBSCRIPTION_TTL:
|
||||||
|
msg = _(u'The TTL for a subscription '
|
||||||
|
'must be at least {0} seconds long.')
|
||||||
|
raise ValidationFailed(msg, MIN_SUBSCRIPTION_TTL)
|
||||||
|
|
||||||
|
# NOTE(flwang): By this change, technically, user can set a very
|
||||||
|
# big TTL so as to get a very long subscription.
|
||||||
|
now = timeutils.utcnow_ts()
|
||||||
|
now_dt = datetime.datetime.utcfromtimestamp(now)
|
||||||
|
msg = _(u'The TTL seconds for a subscription plus current time'
|
||||||
|
' must be less than {0}.')
|
||||||
|
try:
|
||||||
|
# NOTE(flwang): If below expression works, then we believe the
|
||||||
|
# ttl is acceptable otherwise it exceeds the max time of
|
||||||
|
# python.
|
||||||
|
now_dt + datetime.timedelta(seconds=ttl)
|
||||||
|
except OverflowError:
|
||||||
|
raise ValidationFailed(msg, datetime.datetime.max)
|
||||||
|
|
||||||
def subscription_listing(self, limit=None, **kwargs):
|
def subscription_listing(self, limit=None, **kwargs):
|
||||||
"""Restrictions involving a list of subscriptions.
|
"""Restrictions involving a list of subscriptions.
|
||||||
|
|
||||||
|
@ -99,7 +99,8 @@ def public_endpoints(driver, conf):
|
|||||||
# Subscription Endpoints
|
# Subscription Endpoints
|
||||||
('/queues/{queue_name}/subscriptions',
|
('/queues/{queue_name}/subscriptions',
|
||||||
subscriptions.CollectionResource(driver._validate,
|
subscriptions.CollectionResource(driver._validate,
|
||||||
subscription_controller)),
|
subscription_controller,
|
||||||
|
defaults.subscription_ttl)),
|
||||||
|
|
||||||
('/queues/{queue_name}/subscriptions/{subscription_id}',
|
('/queues/{queue_name}/subscriptions/{subscription_id}',
|
||||||
subscriptions.ItemResource(driver._validate,
|
subscriptions.ItemResource(driver._validate,
|
||||||
|
@ -105,11 +105,14 @@ class ItemResource(object):
|
|||||||
|
|
||||||
class CollectionResource(object):
|
class CollectionResource(object):
|
||||||
|
|
||||||
__slots__ = ('_subscription_controller', '_validate')
|
__slots__ = ('_subscription_controller', '_validate',
|
||||||
|
'_default_subscription_ttl')
|
||||||
|
|
||||||
def __init__(self, validate, subscription_controller):
|
def __init__(self, validate, subscription_controller,
|
||||||
|
default_subscription_ttl):
|
||||||
self._subscription_controller = subscription_controller
|
self._subscription_controller = subscription_controller
|
||||||
self._validate = validate
|
self._validate = validate
|
||||||
|
self._default_subscription_ttl = default_subscription_ttl
|
||||||
|
|
||||||
@decorators.TransportLog("Subscription collection")
|
@decorators.TransportLog("Subscription collection")
|
||||||
@acl.enforce("subscription:get_all")
|
@acl.enforce("subscription:get_all")
|
||||||
@ -167,8 +170,8 @@ class CollectionResource(object):
|
|||||||
try:
|
try:
|
||||||
self._validate.subscription_posting(document)
|
self._validate.subscription_posting(document)
|
||||||
subscriber = document['subscriber']
|
subscriber = document['subscriber']
|
||||||
ttl = int(document['ttl'])
|
ttl = document.get('ttl', self._default_subscription_ttl)
|
||||||
options = document['options']
|
options = document.get('options', {})
|
||||||
created = self._subscription_controller.create(queue_name,
|
created = self._subscription_controller.create(queue_name,
|
||||||
subscriber,
|
subscriber,
|
||||||
ttl,
|
ttl,
|
||||||
|
Loading…
Reference in New Issue
Block a user