Merge pull request #525 from datastax/480

PYTHON-480 - cython datetime rounding
This commit is contained in:
Adam Holmberg
2016-03-24 08:48:48 -05:00
3 changed files with 56 additions and 3 deletions

View File

@@ -17,6 +17,8 @@ Duplicate module of util.py, with some accelerated functions
used for deserialization.
"""
from libc.math cimport modf, round, fabs
from cpython.datetime cimport (
timedelta_new,
# cdef inline object timedelta_new(int days, int seconds, int useconds)
@@ -44,6 +46,17 @@ cdef datetime_from_timestamp(double timestamp):
cdef int days = <int> (timestamp / DAY_IN_SECONDS)
cdef int64_t days_in_seconds = (<int64_t> days) * DAY_IN_SECONDS
cdef int seconds = <int> (timestamp - days_in_seconds)
cdef int microseconds = <int> ((timestamp - days_in_seconds - seconds) * 1000000)
cdef double tmp
cdef double micros_left = modf(timestamp, &tmp) * 1000000.
micros_left = modf(micros_left, &tmp)
cdef int microseconds = <int> tmp
# rounding to emulate fp math in delta_new
cdef int x_odd
tmp = round(micros_left)
if fabs(tmp - micros_left) == 0.5:
x_odd = microseconds & 1
tmp = 2.0 * round((micros_left + x_odd) * 0.5) - x_odd
microseconds += <int>tmp
return DATETIME_EPOC + timedelta_new(days, seconds, microseconds)

View File

@@ -26,3 +26,7 @@ class TypesTest(unittest.TestCase):
@cythontest
def test_datetype(self):
types_testhelper.test_datetype(self.assertEqual)
@cythontest
def test_date_side_by_side(self):
types_testhelper.test_date_side_by_side(self.assertEqual)

View File

@@ -12,9 +12,9 @@
# See the License for the specific language governing permissions and
# limitations under the License.
import time
import calendar
import datetime
import time
include '../../../cassandra/ioutils.pyx'
@@ -70,3 +70,39 @@ def test_datetype(assert_equal):
# Large date overflow (PYTHON-452)
expected = 2177403010.123
assert_equal(deserialize(expected), datetime.datetime(2038, 12, 31, 10, 10, 10, 123000))
def test_date_side_by_side(assert_equal):
# Test pure python and cython date deserialization side-by-side
# This is meant to detect inconsistent rounding or conversion (PYTHON-480 for example)
# The test covers the full range of time deserializable in Python. It bounds through
# the range in factors of two to cover floating point scale. At each bound it sweeps
# all combinations of fractional seconds to verify rounding
cdef BytesIOReader reader
cdef Buffer buf
cdef Deserializer cython_deserializer = find_deserializer(DateType)
import time
def verify_time(ms):
blob = DateType.serialize(ms, 0)
bior = BytesIOReader(blob)
buf.ptr = bior.read()
buf.size = bior.size
cython_deserialized = from_binary(cython_deserializer, &buf, 0)
python_deserialized = DateType.deserialize(blob, 0)
assert_equal(cython_deserialized, python_deserialized)
# min -> 0
x = int(calendar.timegm(datetime.datetime(1, 1, 1).utctimetuple()) * 1000)
while x < -1: # for some reason -1 // 2 == -1 so we can't wait for zero
for y in range(1000):
verify_time(x + y)
x //= 2
# max -> 0
x = int(calendar.timegm(datetime.datetime(9999, 12, 31, 23, 59, 59, 999999).utctimetuple()) * 1000)
while x:
for ms in range(1000):
verify_time(x - ms)
x //= 2