Fix the TTL issue of subscriptions for MongoDB

Currently, the TTL of subscriptons doesn't work. That said, after
the ttl seconds, the subscription can't be deleted. This patch
also adds a default TTL for subscription. This patch fixes the bug
and a functional test will be added in Zaqar client.

APIImpact
DocImpact

Partial-Bug: #1529168

Change-Id: I18e6dd3833f11692775b1285365afbced5725249
This commit is contained in:
Fei Long Wang 2016-01-06 10:14:57 +13:00 committed by Eva Balycheva
parent dd016c75c9
commit 1d122b1671
7 changed files with 75 additions and 13 deletions

View File

@ -58,7 +58,7 @@ class RequestSchema(v1_1.RequestSchema):
'ttl': {'type': 'integer'},
'options': {'type': 'object'},
},
'required': ['queue_name', 'ttl'],
'required': ['queue_name', ],
}
},
'required': ['action', 'headers', 'body']

View File

@ -11,6 +11,7 @@
# 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 datetime
from oslo_utils import timeutils
import pymongo.errors
@ -29,6 +30,11 @@ SUBSCRIPTIONS_INDEX = [
('p', 1),
]
# For removing expired subscriptions
TTL_INDEX_FIELDS = [
('e', 1),
]
class SubscriptionController(base.Subscription):
"""Implements subscription resource operations using MongoDB.
@ -49,6 +55,14 @@ class SubscriptionController(base.Subscription):
self._collection = self.driver.subscriptions_database.subscriptions
self._queue_ctrl = self.driver.queue_controller
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
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):
source = queue
now = timeutils.utcnow_ts()
ttl = int(ttl)
expires = now + ttl
now_dt = datetime.datetime.utcfromtimestamp(now)
expires = now_dt + datetime.timedelta(seconds=ttl)
if not self._queue_ctrl.exists(source, project):
raise errors.QueueDoesNotExist(source, project)

View File

@ -97,4 +97,19 @@ class TestValidation(base.V2Base):
self.project_id,
body='{"timespace": "Shangri-la"}',
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)

View File

@ -33,7 +33,9 @@ _RESOURCE_DEFAULTS = (
cfg.IntOpt('default_claim_ttl', default=300,
help=('Defines how long a message will be in claimed state.')),
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'
@ -66,6 +68,10 @@ class ResourceDefaults(object):
def claim_grace(self):
return self._defaults.default_claim_grace
@property
def subscription_ttl(self):
return self._defaults.default_subscription_ttl
@six.add_metaclass(abc.ABCMeta)
class DriverBase(object):

View File

@ -14,9 +14,11 @@
# See the License for the specific language governing permissions and
# limitations under the License.
import datetime
import re
from oslo_config import cfg
from oslo_utils import timeutils
import six
from zaqar.i18n import _
@ -24,6 +26,7 @@ from zaqar.i18n import _
MIN_MESSAGE_TTL = 60
MIN_CLAIM_TTL = 60
MIN_CLAIM_GRACE = 60
MIN_SUBSCRIPTION_TTL = 60
_TRANSPORT_LIMITS_OPTIONS = (
cfg.IntOpt('max_queues_per_page', default=20,
@ -295,7 +298,7 @@ class Validator(object):
:param subscription: dict of subscription
:raises: ValidationFailed if the subscription is invalid.
"""
for p in ('subscriber', 'ttl', 'options'):
for p in ('subscriber',):
if p not in subscription.keys():
raise ValidationFailed(_(u'Missing parameter %s in body.') % p)
@ -329,10 +332,30 @@ class Validator(object):
raise ValidationFailed(msg)
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.')
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):
"""Restrictions involving a list of subscriptions.

View File

@ -99,7 +99,8 @@ def public_endpoints(driver, conf):
# Subscription Endpoints
('/queues/{queue_name}/subscriptions',
subscriptions.CollectionResource(driver._validate,
subscription_controller)),
subscription_controller,
defaults.subscription_ttl)),
('/queues/{queue_name}/subscriptions/{subscription_id}',
subscriptions.ItemResource(driver._validate,

View File

@ -105,11 +105,14 @@ class ItemResource(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._validate = validate
self._default_subscription_ttl = default_subscription_ttl
@decorators.TransportLog("Subscription collection")
@acl.enforce("subscription:get_all")
@ -168,8 +171,8 @@ class CollectionResource(object):
try:
self._validate.subscription_posting(document)
subscriber = document['subscriber']
ttl = int(document['ttl'])
options = document['options']
ttl = document.get('ttl', self._default_subscription_ttl)
options = document.get('options', {})
created = self._subscription_controller.create(queue_name,
subscriber,
ttl,