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:
parent
fb7eb0718a
commit
1e380473cd
2 changed files with 40 additions and 38 deletions
|
@ -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):
|
||||||
"""
|
"""
|
||||||
|
|
|
@ -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:
|
||||||
|
|
Loading…
Add table
Reference in a new issue