Datetime instances from SQLAlchemy now all contain timezone.

One of the complex problems in storyboard has been that none of our
datetime instances contain a timezone, making comparison operations
guesswork at best. This is partially exacerbated by oslo.db's
TimestampMixin, which ensures created_at and updated_at parameters
are set, but does not inform the instance that its underlying
assumption is to store datetime in UTC.

This patch addresses this, by creating our own UTCDateTime SQLAlchemy
type decorator and applying it everywhere a datetime is needed.

Change-Id: Iea89661e4d370f0e8af93d2c033b4c44723a39a4
This commit is contained in:
Michael Krotscheck 2015-02-11 13:29:38 -08:00 committed by Aleksey Ripinen
parent 552f2848b3
commit db53c6b2c5
11 changed files with 94 additions and 35 deletions

View File

@ -14,6 +14,7 @@
# limitations under the License. # limitations under the License.
import datetime import datetime
import pytz
from oauthlib.oauth2 import RequestValidator from oauthlib.oauth2 import RequestValidator
from oauthlib.oauth2 import WebApplicationServer from oauthlib.oauth2 import WebApplicationServer
@ -111,7 +112,7 @@ class SkeletonValidator(RequestValidator):
email = request._params["openid.sreg.email"] email = request._params["openid.sreg.email"]
full_name = request._params["openid.sreg.fullname"] full_name = request._params["openid.sreg.fullname"]
username = request._params["openid.sreg.nickname"] username = request._params["openid.sreg.nickname"]
last_login = datetime.datetime.utcnow() last_login = datetime.datetime.now(pytz.utc)
user = user_api.user_get_by_openid(openid) user = user_api.user_get_by_openid(openid)
user_dict = {"full_name": full_name, user_dict = {"full_name": full_name,
@ -212,7 +213,7 @@ class SkeletonValidator(RequestValidator):
access_token_values = { access_token_values = {
"access_token": token["access_token"], "access_token": token["access_token"],
"expires_in": token["expires_in"], "expires_in": token["expires_in"],
"expires_at": datetime.datetime.utcnow() + datetime.timedelta( "expires_at": datetime.datetime.now(pytz.utc) + datetime.timedelta(
seconds=token["expires_in"]), seconds=token["expires_in"]),
"user_id": user_id "user_id": user_id
} }
@ -227,7 +228,7 @@ class SkeletonValidator(RequestValidator):
"refresh_token": token["refresh_token"], "refresh_token": token["refresh_token"],
"user_id": user_id, "user_id": user_id,
"expires_in": refresh_expires_in, "expires_in": refresh_expires_in,
"expires_at": datetime.datetime.utcnow() + datetime.timedelta( "expires_at": datetime.datetime.now(pytz.utc) + datetime.timedelta(
seconds=refresh_expires_in), seconds=refresh_expires_in),
} }
auth_api.refresh_token_save(refresh_token_values) auth_api.refresh_token_save(refresh_token_values)
@ -271,7 +272,7 @@ class SkeletonValidator(RequestValidator):
if not refresh_token_entry: if not refresh_token_entry:
return False return False
if datetime.datetime.utcnow() > refresh_token_entry.expires_at: if datetime.datetime.now(pytz.utc) > refresh_token_entry.expires_at:
auth_api.refresh_token_delete(refresh_token) auth_api.refresh_token_delete(refresh_token)
return False return False

View File

@ -14,6 +14,7 @@
# limitations under the License. # limitations under the License.
import datetime import datetime
import pytz
from storyboard.db.api import base as api_base from storyboard.db.api import base as api_base
from storyboard.db import models from storyboard.db import models
@ -37,7 +38,7 @@ def is_valid(access_token):
if not token: if not token:
return False return False
if datetime.datetime.utcnow() > token.expires_at: if datetime.datetime.now(tz=pytz.utc) > token.expires_at:
access_token_delete(token.id) access_token_delete(token.id)
return False return False
@ -75,9 +76,9 @@ def access_token_get_count(**kwargs):
def access_token_create(values): def access_token_create(values):
# Update the expires_at date. # Update the expires_at date.
values['created_at'] = datetime.datetime.utcnow() values['created_at'] = datetime.datetime.now(pytz.utc)
values['expires_at'] = datetime.datetime.utcnow() + datetime.timedelta( values['expires_at'] = datetime.datetime.now(pytz.utc) + datetime \
seconds=values['expires_in']) .timedelta(seconds=values['expires_in'])
return api_base.entity_create(models.AccessToken, values) return api_base.entity_create(models.AccessToken, values)

View File

@ -0,0 +1,41 @@
# Copyright 2015 Hewlett-Packard Development Company, L.P.
#
# 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 sqlalchemy.types import TypeDecorator, DateTime
import pytz
class UTCDateTime(TypeDecorator):
"""This decorator ensures that timezones will always remain attached to
datetime fields that are written to/from the database, since mysql does
NOT support timezones in its datetime fields. In the case that a
value is provided without a timezone, it will raise an exception to draw
attention to itself: Engineers should always work with timezoned
datetime instances.
"""
impl = DateTime
def process_bind_param(self, value, engine):
if value is not None:
# If the value doesn't have a timezone, raise an exception.
if not value.tzinfo:
raise RuntimeError("Datetime value without timezone provided,"
" please provide a timezone.")
# Convert to UTC, then scrub the timezone for storing in MySQL.
return value.astimezone(pytz.utc).replace(tzinfo=None)
def process_result_value(self, value, engine):
if value is not None:
return value.replace(tzinfo=pytz.utc)

View File

@ -17,11 +17,15 @@
SQLAlchemy Models for storing storyboard SQLAlchemy Models for storing storyboard
""" """
import datetime
import pytz
import six
import six.moves.urllib.parse as urlparse
from oslo.config import cfg from oslo.config import cfg
from oslo.db.sqlalchemy import models from oslo.db.sqlalchemy import models
from sqlalchemy import Boolean from sqlalchemy import Boolean
from sqlalchemy import Column from sqlalchemy import Column
from sqlalchemy import DateTime
from sqlalchemy.dialects.mysql import MEDIUMTEXT from sqlalchemy.dialects.mysql import MEDIUMTEXT
from sqlalchemy import Enum from sqlalchemy import Enum
from sqlalchemy.ext import declarative from sqlalchemy.ext import declarative
@ -38,8 +42,7 @@ from sqlalchemy import Unicode
from sqlalchemy import UnicodeText from sqlalchemy import UnicodeText
from sqlalchemy_fulltext import FullText from sqlalchemy_fulltext import FullText
import six from storyboard.db.decorators import UTCDateTime
import six.moves.urllib.parse as urlparse
CONF = cfg.CONF CONF = cfg.CONF
@ -71,7 +74,18 @@ class IdMixin(object):
id = Column(Integer, primary_key=True) id = Column(Integer, primary_key=True)
class StoriesBase(models.TimestampMixin, class UTCTimestampMixin(object):
"""A Database model mixin that automatically manages our creation and
updating timestamps. This mixin was copied from oslo.db, and adapted to
use our own internal UTCDateTime type decorator.
"""
created_at = Column(UTCDateTime,
default=lambda: datetime.datetime.now(tz=pytz.utc))
updated_at = Column(UTCDateTime,
onupdate=lambda: datetime.datetime.now(tz=pytz.utc))
class StoriesBase(UTCTimestampMixin,
IdMixin, IdMixin,
models.ModelBase): models.ModelBase):
metadata = None metadata = None
@ -134,7 +148,7 @@ class User(FullText, ModelBuilder, Base):
is_staff = Column(Boolean, default=False) is_staff = Column(Boolean, default=False)
is_active = Column(Boolean, default=True) is_active = Column(Boolean, default=True)
is_superuser = Column(Boolean, default=False) is_superuser = Column(Boolean, default=False)
last_login = Column(DateTime) last_login = Column(UTCDateTime)
teams = relationship("Team", secondary="team_membership") teams = relationship("Team", secondary="team_membership")
permissions = relationship("Permission", secondary="user_permissions") permissions = relationship("Permission", secondary="user_permissions")
enable_login = Column(Boolean, default=True) enable_login = Column(Boolean, default=True)
@ -316,7 +330,7 @@ class AccessToken(ModelBuilder, Base):
access_token = Column(Unicode(CommonLength.top_middle_length), access_token = Column(Unicode(CommonLength.top_middle_length),
nullable=False) nullable=False)
expires_in = Column(Integer, nullable=False) expires_in = Column(Integer, nullable=False)
expires_at = Column(DateTime, nullable=False) expires_at = Column(UTCDateTime, nullable=False)
refresh_tokens = relationship("RefreshToken", refresh_tokens = relationship("RefreshToken",
cascade="save-update, merge, delete", cascade="save-update, merge, delete",
passive_updates=False, passive_updates=False,
@ -331,7 +345,7 @@ class RefreshToken(ModelBuilder, Base):
refresh_token = Column(Unicode(CommonLength.top_middle_length), refresh_token = Column(Unicode(CommonLength.top_middle_length),
nullable=False) nullable=False)
expires_in = Column(Integer, nullable=False) expires_in = Column(Integer, nullable=False)
expires_at = Column(DateTime, nullable=False) expires_at = Column(UTCDateTime, nullable=False)
def _story_build_summary_query(): def _story_build_summary_query():

View File

@ -150,12 +150,12 @@ class LaunchpadWriter(object):
""" """
if hasattr(bug, 'date_created'): if hasattr(bug, 'date_created'):
created_at = bug.date_created.strftime('%Y-%m-%d %H:%M:%S') created_at = bug.date_created
else: else:
created_at = None created_at = None
if hasattr(bug, 'date_last_updated'): if hasattr(bug, 'date_last_updated'):
updated_at = bug.date_last_updated.strftime('%Y-%m-%d %H:%M:%S') updated_at = bug.date_last_updated
else: else:
updated_at = None updated_at = None
@ -263,8 +263,7 @@ class LaunchpadWriter(object):
for i in range(current_count, desired_count): for i in range(current_count, desired_count):
print '- Importing comment %s of %s' % (i + 1, desired_count) print '- Importing comment %s of %s' % (i + 1, desired_count)
message = bug.messages[i] message = bug.messages[i]
message_created_at = message.date_created \ message_created_at = message.date_created
.strftime('%Y-%m-%d %H:%M:%S')
message_owner = self.write_user(message.owner) message_owner = self.write_user(message.owner)
comment = comments_api.comment_create({ comment = comments_api.comment_create({

View File

@ -109,7 +109,7 @@ class CronPluginBase(plugin_base.PluginBase):
lr_file = os.path.join(cron_directory, plugin_name) lr_file = os.path.join(cron_directory, plugin_name)
now = pytz.utc.localize(datetime.datetime.utcnow()) now = datetime.datetime.now(pytz.utc)
start_time = self._get_file_mtime(path=lr_file) start_time = self._get_file_mtime(path=lr_file)
end_time = self._get_file_mtime(path=lr_file, end_time = self._get_file_mtime(path=lr_file,

View File

@ -14,6 +14,7 @@
from datetime import datetime from datetime import datetime
from datetime import timedelta from datetime import timedelta
import pytz
import storyboard.db.api.base as api_base import storyboard.db.api.base as api_base
from storyboard.db.models import AccessToken from storyboard.db.models import AccessToken
@ -49,7 +50,7 @@ class TokenCleaner(CronPluginBase):
:param end_time: The current timestamp. :param end_time: The current timestamp.
""" """
# Calculate last week. # Calculate last week.
lastweek = datetime.utcnow() - timedelta(weeks=1) lastweek = datetime.now(pytz.utc) - timedelta(weeks=1)
# Build the query. # Build the query.
query = api_base.model_query(AccessToken.id) query = api_base.model_query(AccessToken.id)

View File

@ -14,6 +14,7 @@
# limitations under the License. # limitations under the License.
from datetime import datetime from datetime import datetime
import pytz
from storyboard.db.api import access_tokens from storyboard.db.api import access_tokens
from storyboard.db.api import users from storyboard.db.api import users
@ -29,7 +30,7 @@ class TokenTest(base.BaseDbTestCase):
"access_token": u'an_access_token', "access_token": u'an_access_token',
"refresh_token": u'a_refresh_token', "refresh_token": u'a_refresh_token',
"expires_in": 3600, "expires_in": 3600,
"expires_at": datetime.utcnow(), "expires_at": datetime.now(pytz.utc),
"user_id": 1 "user_id": 1
} }

View File

@ -13,6 +13,7 @@
# under the License. # under the License.
import datetime import datetime
import pytz
import storyboard.common.event_types as event import storyboard.common.event_types as event
from storyboard.db.api import base as db from storyboard.db.api import base as db
@ -31,7 +32,7 @@ def load():
"""Load a batch of useful data into the database that our tests can work """Load a batch of useful data into the database that our tests can work
with. with.
""" """
now = datetime.datetime.utcnow() now = datetime.datetime.now(tz=pytz.utc)
expires_at = now + datetime.timedelta(seconds=3600) expires_at = now + datetime.timedelta(seconds=3600)
expired_at = now + datetime.timedelta(seconds=-3600) expired_at = now + datetime.timedelta(seconds=-3600)
@ -63,22 +64,22 @@ def load():
user_id=1, user_id=1,
access_token='valid_superuser_token', access_token='valid_superuser_token',
expires_in=3600, expires_in=3600,
expires_at=expires_at.strftime('%Y-%m-%d %H:%M:%S')), expires_at=expires_at),
AccessToken( AccessToken(
user_id=1, user_id=1,
access_token='expired_superuser_token', access_token='expired_superuser_token',
expires_in=3600, expires_in=3600,
expires_at=expired_at.strftime('%Y-%m-%d %H:%M:%S')), expires_at=expired_at),
AccessToken( AccessToken(
user_id=2, user_id=2,
access_token='valid_user_token', access_token='valid_user_token',
expires_in=3600, expires_in=3600,
expires_at=expires_at.strftime('%Y-%m-%d %H:%M:%S')), expires_at=expires_at),
AccessToken( AccessToken(
user_id=2, user_id=2,
access_token='expired_user_token', access_token='expired_user_token',
expires_in=3600, expires_in=3600,
expires_at=expired_at.strftime('%Y-%m-%d %H:%M:%S')) expires_at=expired_at)
]) ])
# Create some test projects. # Create some test projects.

View File

@ -105,8 +105,7 @@ class TestCronPluginBase(base.WorkingDirTestCase):
# Current timestamp, remove microseconds so that we don't run into # Current timestamp, remove microseconds so that we don't run into
# execution time delay problems. # execution time delay problems.
now = pytz.utc.localize(datetime.datetime.utcnow()) \ now = datetime.datetime.now(pytz.utc).replace(microsecond=0)
.replace(microsecond=0)
# Check the plugin's params. # Check the plugin's params.
self.assertEqual(last_run_date, plugin.last_invocation_parameters[0]) self.assertEqual(last_run_date, plugin.last_invocation_parameters[0])

View File

@ -14,6 +14,7 @@
from datetime import datetime from datetime import datetime
from datetime import timedelta from datetime import timedelta
import pytz
from oslo.config import cfg from oslo.config import cfg
import storyboard.db.api.base as db_api import storyboard.db.api.base as db_api
@ -67,16 +68,16 @@ class TestTokenCleaner(db_base.BaseDbTestCase,
new_access_tokens = [] new_access_tokens = []
new_refresh_tokens = [] new_refresh_tokens = []
for i in range(0, 100): for i in range(0, 100):
created_at = datetime.utcnow() - timedelta(days=i) created_at = datetime.now(pytz.utc) - timedelta(days=i)
expires_in = (60 * 60 * 24) - 5 # Minus five seconds, see above. expires_in = (60 * 60 * 24) - 5 # Minus five seconds, see above.
expires_at = created_at + timedelta(seconds=expires_in) expires_at = created_at + timedelta(seconds=expires_in)
new_access_tokens.append( new_access_tokens.append(
AccessToken( AccessToken(
user_id=1, user_id=1,
created_at=created_at.strftime('%Y-%m-%d %H:%M:%S'), created_at=created_at,
expires_in=expires_in, expires_in=expires_in,
expires_at=expires_at.strftime('%Y-%m-%d %H:%M:%S'), expires_at=expires_at,
access_token='test_token_%s' % (i,)) access_token='test_token_%s' % (i,))
) )
new_access_tokens = load_data(new_access_tokens) new_access_tokens = load_data(new_access_tokens)