mirror of
https://github.com/beetbox/beets.git
synced 2025-12-06 08:39:17 +01:00
322 lines
11 KiB
Python
322 lines
11 KiB
Python
# This file is part of beets.
|
|
# Copyright 2016, Adrian Sampson.
|
|
#
|
|
# Permission is hereby granted, free of charge, to any person obtaining
|
|
# a copy of this software and associated documentation files (the
|
|
# "Software"), to deal in the Software without restriction, including
|
|
# without limitation the rights to use, copy, modify, merge, publish,
|
|
# distribute, sublicense, and/or sell copies of the Software, and to
|
|
# permit persons to whom the Software is furnished to do so, subject to
|
|
# the following conditions:
|
|
#
|
|
# The above copyright notice and this permission notice shall be
|
|
# included in all copies or substantial portions of the Software.
|
|
|
|
"""Test for dbcore's date-based queries.
|
|
"""
|
|
|
|
import time
|
|
import unittest
|
|
from datetime import datetime, timedelta
|
|
|
|
import pytest
|
|
|
|
from beets.dbcore.query import (
|
|
DateInterval,
|
|
DateQuery,
|
|
InvalidQueryArgumentValueError,
|
|
_parse_periods,
|
|
)
|
|
from beets.test.helper import ItemInDBTestCase
|
|
|
|
|
|
def _date(string):
|
|
return datetime.strptime(string, "%Y-%m-%dT%H:%M:%S")
|
|
|
|
|
|
def _datepattern(datetimedate):
|
|
return datetimedate.strftime("%Y-%m-%dT%H:%M:%S")
|
|
|
|
|
|
class DateIntervalTest(unittest.TestCase):
|
|
def test_year_precision_intervals(self):
|
|
self.assertContains("2000..2001", "2000-01-01T00:00:00")
|
|
self.assertContains("2000..2001", "2001-06-20T14:15:16")
|
|
self.assertContains("2000..2001", "2001-12-31T23:59:59")
|
|
self.assertExcludes("2000..2001", "1999-12-31T23:59:59")
|
|
self.assertExcludes("2000..2001", "2002-01-01T00:00:00")
|
|
|
|
self.assertContains("2000..", "2000-01-01T00:00:00")
|
|
self.assertContains("2000..", "2099-10-11T00:00:00")
|
|
self.assertExcludes("2000..", "1999-12-31T23:59:59")
|
|
|
|
self.assertContains("..2001", "2001-12-31T23:59:59")
|
|
self.assertExcludes("..2001", "2002-01-01T00:00:00")
|
|
|
|
self.assertContains("-1d..1d", _datepattern(datetime.now()))
|
|
self.assertExcludes("-2d..-1d", _datepattern(datetime.now()))
|
|
|
|
def test_day_precision_intervals(self):
|
|
self.assertContains("2000-06-20..2000-06-20", "2000-06-20T00:00:00")
|
|
self.assertContains("2000-06-20..2000-06-20", "2000-06-20T10:20:30")
|
|
self.assertContains("2000-06-20..2000-06-20", "2000-06-20T23:59:59")
|
|
self.assertExcludes("2000-06-20..2000-06-20", "2000-06-19T23:59:59")
|
|
self.assertExcludes("2000-06-20..2000-06-20", "2000-06-21T00:00:00")
|
|
|
|
def test_month_precision_intervals(self):
|
|
self.assertContains("1999-12..2000-02", "1999-12-01T00:00:00")
|
|
self.assertContains("1999-12..2000-02", "2000-02-15T05:06:07")
|
|
self.assertContains("1999-12..2000-02", "2000-02-29T23:59:59")
|
|
self.assertExcludes("1999-12..2000-02", "1999-11-30T23:59:59")
|
|
self.assertExcludes("1999-12..2000-02", "2000-03-01T00:00:00")
|
|
|
|
def test_hour_precision_intervals(self):
|
|
# test with 'T' separator
|
|
self.assertExcludes(
|
|
"2000-01-01T12..2000-01-01T13", "2000-01-01T11:59:59"
|
|
)
|
|
self.assertContains(
|
|
"2000-01-01T12..2000-01-01T13", "2000-01-01T12:00:00"
|
|
)
|
|
self.assertContains(
|
|
"2000-01-01T12..2000-01-01T13", "2000-01-01T12:30:00"
|
|
)
|
|
self.assertContains(
|
|
"2000-01-01T12..2000-01-01T13", "2000-01-01T13:30:00"
|
|
)
|
|
self.assertContains(
|
|
"2000-01-01T12..2000-01-01T13", "2000-01-01T13:59:59"
|
|
)
|
|
self.assertExcludes(
|
|
"2000-01-01T12..2000-01-01T13", "2000-01-01T14:00:00"
|
|
)
|
|
self.assertExcludes(
|
|
"2000-01-01T12..2000-01-01T13", "2000-01-01T14:30:00"
|
|
)
|
|
|
|
# test non-range query
|
|
self.assertContains("2008-12-01T22", "2008-12-01T22:30:00")
|
|
self.assertExcludes("2008-12-01T22", "2008-12-01T23:30:00")
|
|
|
|
def test_minute_precision_intervals(self):
|
|
self.assertExcludes(
|
|
"2000-01-01T12:30..2000-01-01T12:31", "2000-01-01T12:29:59"
|
|
)
|
|
self.assertContains(
|
|
"2000-01-01T12:30..2000-01-01T12:31", "2000-01-01T12:30:00"
|
|
)
|
|
self.assertContains(
|
|
"2000-01-01T12:30..2000-01-01T12:31", "2000-01-01T12:30:30"
|
|
)
|
|
self.assertContains(
|
|
"2000-01-01T12:30..2000-01-01T12:31", "2000-01-01T12:31:59"
|
|
)
|
|
self.assertExcludes(
|
|
"2000-01-01T12:30..2000-01-01T12:31", "2000-01-01T12:32:00"
|
|
)
|
|
|
|
def test_second_precision_intervals(self):
|
|
self.assertExcludes(
|
|
"2000-01-01T12:30:50..2000-01-01T12:30:55", "2000-01-01T12:30:49"
|
|
)
|
|
self.assertContains(
|
|
"2000-01-01T12:30:50..2000-01-01T12:30:55", "2000-01-01T12:30:50"
|
|
)
|
|
self.assertContains(
|
|
"2000-01-01T12:30:50..2000-01-01T12:30:55", "2000-01-01T12:30:55"
|
|
)
|
|
self.assertExcludes(
|
|
"2000-01-01T12:30:50..2000-01-01T12:30:55", "2000-01-01T12:30:56"
|
|
)
|
|
|
|
def test_unbounded_endpoints(self):
|
|
self.assertContains("..", date=datetime.max)
|
|
self.assertContains("..", date=datetime.min)
|
|
self.assertContains("..", "1000-01-01T00:00:00")
|
|
|
|
def assertContains( # noqa
|
|
self, interval_pattern, date_pattern=None, date=None
|
|
):
|
|
if date is None:
|
|
date = _date(date_pattern)
|
|
(start, end) = _parse_periods(interval_pattern)
|
|
interval = DateInterval.from_periods(start, end)
|
|
assert interval.contains(date)
|
|
|
|
def assertExcludes(self, interval_pattern, date_pattern): # noqa
|
|
date = _date(date_pattern)
|
|
(start, end) = _parse_periods(interval_pattern)
|
|
interval = DateInterval.from_periods(start, end)
|
|
assert not interval.contains(date)
|
|
|
|
|
|
def _parsetime(s):
|
|
return time.mktime(datetime.strptime(s, "%Y-%m-%d %H:%M").timetuple())
|
|
|
|
|
|
class DateQueryTest(ItemInDBTestCase):
|
|
def setUp(self):
|
|
super().setUp()
|
|
self.i.added = _parsetime("2013-03-30 22:21")
|
|
self.i.store()
|
|
|
|
def test_single_month_match_fast(self):
|
|
query = DateQuery("added", "2013-03")
|
|
matched = self.lib.items(query)
|
|
assert len(matched) == 1
|
|
|
|
def test_single_month_nonmatch_fast(self):
|
|
query = DateQuery("added", "2013-04")
|
|
matched = self.lib.items(query)
|
|
assert len(matched) == 0
|
|
|
|
def test_single_month_match_slow(self):
|
|
query = DateQuery("added", "2013-03")
|
|
assert query.match(self.i)
|
|
|
|
def test_single_month_nonmatch_slow(self):
|
|
query = DateQuery("added", "2013-04")
|
|
assert not query.match(self.i)
|
|
|
|
def test_single_day_match_fast(self):
|
|
query = DateQuery("added", "2013-03-30")
|
|
matched = self.lib.items(query)
|
|
assert len(matched) == 1
|
|
|
|
def test_single_day_nonmatch_fast(self):
|
|
query = DateQuery("added", "2013-03-31")
|
|
matched = self.lib.items(query)
|
|
assert len(matched) == 0
|
|
|
|
|
|
class DateQueryTestRelative(ItemInDBTestCase):
|
|
def setUp(self):
|
|
super().setUp()
|
|
|
|
# We pick a date near a month changeover, which can reveal some time
|
|
# zone bugs.
|
|
self._now = datetime(2017, 12, 31, 22, 55, 4, 101332)
|
|
|
|
self.i.added = _parsetime(self._now.strftime("%Y-%m-%d %H:%M"))
|
|
self.i.store()
|
|
|
|
def test_single_month_match_fast(self):
|
|
query = DateQuery("added", self._now.strftime("%Y-%m"))
|
|
matched = self.lib.items(query)
|
|
assert len(matched) == 1
|
|
|
|
def test_single_month_nonmatch_fast(self):
|
|
query = DateQuery(
|
|
"added", (self._now + timedelta(days=30)).strftime("%Y-%m")
|
|
)
|
|
matched = self.lib.items(query)
|
|
assert len(matched) == 0
|
|
|
|
def test_single_month_match_slow(self):
|
|
query = DateQuery("added", self._now.strftime("%Y-%m"))
|
|
assert query.match(self.i)
|
|
|
|
def test_single_month_nonmatch_slow(self):
|
|
query = DateQuery(
|
|
"added", (self._now + timedelta(days=30)).strftime("%Y-%m")
|
|
)
|
|
assert not query.match(self.i)
|
|
|
|
def test_single_day_match_fast(self):
|
|
query = DateQuery("added", self._now.strftime("%Y-%m-%d"))
|
|
matched = self.lib.items(query)
|
|
assert len(matched) == 1
|
|
|
|
def test_single_day_nonmatch_fast(self):
|
|
query = DateQuery(
|
|
"added", (self._now + timedelta(days=1)).strftime("%Y-%m-%d")
|
|
)
|
|
matched = self.lib.items(query)
|
|
assert len(matched) == 0
|
|
|
|
|
|
class DateQueryTestRelativeMore(ItemInDBTestCase):
|
|
def setUp(self):
|
|
super().setUp()
|
|
self.i.added = _parsetime(datetime.now().strftime("%Y-%m-%d %H:%M"))
|
|
self.i.store()
|
|
|
|
def test_relative(self):
|
|
for timespan in ["d", "w", "m", "y"]:
|
|
query = DateQuery("added", "-4" + timespan + "..+4" + timespan)
|
|
matched = self.lib.items(query)
|
|
assert len(matched) == 1
|
|
|
|
def test_relative_fail(self):
|
|
for timespan in ["d", "w", "m", "y"]:
|
|
query = DateQuery("added", "-2" + timespan + "..-1" + timespan)
|
|
matched = self.lib.items(query)
|
|
assert len(matched) == 0
|
|
|
|
def test_start_relative(self):
|
|
for timespan in ["d", "w", "m", "y"]:
|
|
query = DateQuery("added", "-4" + timespan + "..")
|
|
matched = self.lib.items(query)
|
|
assert len(matched) == 1
|
|
|
|
def test_start_relative_fail(self):
|
|
for timespan in ["d", "w", "m", "y"]:
|
|
query = DateQuery("added", "4" + timespan + "..")
|
|
matched = self.lib.items(query)
|
|
assert len(matched) == 0
|
|
|
|
def test_end_relative(self):
|
|
for timespan in ["d", "w", "m", "y"]:
|
|
query = DateQuery("added", "..+4" + timespan)
|
|
matched = self.lib.items(query)
|
|
assert len(matched) == 1
|
|
|
|
def test_end_relative_fail(self):
|
|
for timespan in ["d", "w", "m", "y"]:
|
|
query = DateQuery("added", "..-4" + timespan)
|
|
matched = self.lib.items(query)
|
|
assert len(matched) == 0
|
|
|
|
|
|
class DateQueryConstructTest(unittest.TestCase):
|
|
def test_long_numbers(self):
|
|
with pytest.raises(InvalidQueryArgumentValueError):
|
|
DateQuery("added", "1409830085..1412422089")
|
|
|
|
def test_too_many_components(self):
|
|
with pytest.raises(InvalidQueryArgumentValueError):
|
|
DateQuery("added", "12-34-56-78")
|
|
|
|
def test_invalid_date_query(self):
|
|
q_list = [
|
|
"2001-01-0a",
|
|
"2001-0a",
|
|
"200a",
|
|
"2001-01-01..2001-01-0a",
|
|
"2001-0a..2001-01",
|
|
"200a..2002",
|
|
"20aa..",
|
|
"..2aa",
|
|
]
|
|
for q in q_list:
|
|
with pytest.raises(InvalidQueryArgumentValueError):
|
|
DateQuery("added", q)
|
|
|
|
def test_datetime_uppercase_t_separator(self):
|
|
date_query = DateQuery("added", "2000-01-01T12")
|
|
assert date_query.interval.start == datetime(2000, 1, 1, 12)
|
|
assert date_query.interval.end == datetime(2000, 1, 1, 13)
|
|
|
|
def test_datetime_lowercase_t_separator(self):
|
|
date_query = DateQuery("added", "2000-01-01t12")
|
|
assert date_query.interval.start == datetime(2000, 1, 1, 12)
|
|
assert date_query.interval.end == datetime(2000, 1, 1, 13)
|
|
|
|
def test_datetime_space_separator(self):
|
|
date_query = DateQuery("added", "2000-01-01 12")
|
|
assert date_query.interval.start == datetime(2000, 1, 1, 12)
|
|
assert date_query.interval.end == datetime(2000, 1, 1, 13)
|
|
|
|
def test_datetime_invalid_separator(self):
|
|
with pytest.raises(InvalidQueryArgumentValueError):
|
|
DateQuery("added", "2000-01-01x12")
|