Refactor get_days_with_logs()

- use NATURAL JOIN in SQL query instead of multiple SELECT via
_build_contact_where

- make code more concise

- update method documentation
This commit is contained in:
Philipp Hörist 2017-07-16 01:14:42 +02:00
parent fb7eb0718a
commit 1e380473cd
2 changed files with 40 additions and 38 deletions

View file

@ -32,6 +32,7 @@ import os
import sys import sys
import time import time
import datetime import datetime
import calendar
import json import json
from collections import namedtuple from collections import namedtuple
from gzip import GzipFile from gzip import GzipFile
@ -766,42 +767,45 @@ class Logger:
return self.con.execute(sql, (*jids, query)).fetchall() return self.con.execute(sql, (*jids, query)).fetchall()
def get_days_with_logs(self, jid, year, month, max_day, account): def get_days_with_logs(self, account, jid, year, month):
""" """
Return the list of days that have logs (not status messages) Request the days in a month where we received messages
for a given `jid`.
:param account: The account
:param jid: The jid for which we request the days
:param year: The year
:param month: The month
returns a list of namedtuples
""" """
try: jids = self._get_family_jids(account, jid)
self.get_jid_id(jid)
except exceptions.PysqliteOperationalError:
# Error trying to create a new jid_id. This means there is no log
return []
days_with_logs = []
where_sql, jid_tuple = self._build_contact_where(account, jid)
# First select all date of month whith logs we want kinds = map(str, [KindConstant.STATUS,
start_of_month = self.get_unix_time_from_date(year, month, 1) KindConstant.GCSTATUS])
seconds_in_a_day = 86400 # 60 * 60 * 24
last_second_of_month = start_of_month + (seconds_in_a_day * max_day) - 1
# Select times and 'floor' them to time 0:00 # Calculate the start and end datetime of the month
# (by dividing, they are integers) date = datetime.datetime(year, month, 1)
# and take only one of the same values (distinct) days = calendar.monthrange(year, month)[1] - 1
# Now we have timestamps of time 0:00 of every day with logs delta = datetime.timedelta(
self.cur.execute(''' days=days, hours=23, minutes=59, seconds=59, microseconds=999999)
SELECT DISTINCT time/(86400)*86400 as time FROM logs
WHERE (%s) sql = """
AND time BETWEEN %d AND %d SELECT DISTINCT
AND kind NOT IN (%d, %d) CAST(strftime('%d', time, 'unixepoch', 'localtime') AS INTEGER)
AS day FROM logs NATURAL JOIN jids WHERE jid IN ({jids})
AND time BETWEEN ? AND ?
AND kind NOT IN ({kinds})
ORDER BY time ORDER BY time
''' % (where_sql, start_of_month, last_second_of_month, """.format(jids=', '.join('?' * len(jids)),
KindConstant.STATUS, KindConstant.GCSTATUS), jid_tuple) kinds=', '.join(kinds))
result = self.cur.fetchall()
# convert timestamps to day of month return self.con.execute(sql, (*jids,
for line in result: date.timestamp(),
days_with_logs[0:0]=[time.gmtime(line.time)[2]] (date + delta).timestamp())).fetchall()
return days_with_logs
def get_last_date_that_has_logs(self, jid, account=None, is_room=False): def get_last_date_that_has_logs(self, jid, account=None, is_room=False):
""" """

View file

@ -357,20 +357,18 @@ class HistoryWindow:
widget.select_day(1) widget.select_day(1)
return return
# in gtk January is 1, in python January is 0,
# I want the second
# first day of month is 1 not 0
widget.clear_marks() widget.clear_marks()
month = gtkgui_helpers.make_gtk_month_python_month(month) month = gtkgui_helpers.make_gtk_month_python_month(month)
days_in_this_month = calendar.monthrange(year, month)[1]
try: try:
log_days = gajim.logger.get_days_with_logs(self.jid, year, month, log_days = gajim.logger.get_days_with_logs(
days_in_this_month, self.account) self.account, self.jid, year, month)
except exceptions.PysqliteOperationalError as e: except exceptions.PysqliteOperationalError as e:
dialogs.ErrorDialog(_('Disk Error'), str(e)) dialogs.ErrorDialog(_('Disk Error'), str(e))
return return
for day in log_days:
widget.mark_day(day) for date in log_days:
widget.mark_day(date.day)
def _get_string_show_from_constant_int(self, show): def _get_string_show_from_constant_int(self, show):
if show == ShowConstant.ONLINE: if show == ShowConstant.ONLINE: