Refactor database migration

- Move database migration to the logger module
- Dont depend on Gajim version for migration use PRAGMA user_version
- Refactoring of some methods in the logger module
- Dont use cursor object, if there is no reason
- Make some attributes and methods private
This commit is contained in:
Philipp Hörist 2018-04-23 20:27:19 +02:00
parent 4c1bfda48d
commit 5ee45b86e9
3 changed files with 244 additions and 296 deletions

0
Logs.db Normal file
View File

View File

@ -1,5 +1,5 @@
# -*- coding:utf-8 -*- # -*- coding:utf-8 -*-
## src/common/logger.py ## gajim/common/logger.py
## ##
## Copyright (C) 2003-2014 Yann Leboulanger <asterix AT lagaule.org> ## Copyright (C) 2003-2014 Yann Leboulanger <asterix AT lagaule.org>
## Copyright (C) 2004-2005 Vincent Hanquez <tab AT snarc.org> ## Copyright (C) 2004-2005 Vincent Hanquez <tab AT snarc.org>
@ -8,6 +8,7 @@
## Copyright (C) 2006-2008 Jean-Marie Traissard <jim AT lapin.org> ## Copyright (C) 2006-2008 Jean-Marie Traissard <jim AT lapin.org>
## Copyright (C) 2007 Tomasz Melcer <liori AT exroot.org> ## Copyright (C) 2007 Tomasz Melcer <liori AT exroot.org>
## Julien Pivotto <roidelapluie AT gmail.com> ## Julien Pivotto <roidelapluie AT gmail.com>
## Copyright (C) 2018 Philipp Hörist <philipp AT hoerist.com>
## ##
## This file is part of Gajim. ## This file is part of Gajim.
## ##
@ -34,11 +35,12 @@ import time
import datetime import datetime
import calendar import calendar
import json import json
import logging
import sqlite3 as sqlite
from collections import namedtuple from collections import namedtuple
from gzip import GzipFile from gzip import GzipFile
from io import BytesIO from io import BytesIO
from gi.repository import GLib from gi.repository import GLib
from enum import IntEnum, unique
from gajim.common import exceptions from gajim.common import exceptions
from gajim.common import app from gajim.common import app
@ -47,72 +49,22 @@ from gajim.common.const import (
JIDConstant, KindConstant, ShowConstant, TypeConstant, JIDConstant, KindConstant, ShowConstant, TypeConstant,
SubscriptionConstant) SubscriptionConstant)
import sqlite3 as sqlite
LOG_DB_PATH = configpaths.get('LOG_DB') LOG_DB_PATH = configpaths.get('LOG_DB')
LOG_DB_FOLDER, LOG_DB_FILE = os.path.split(LOG_DB_PATH) LOG_DB_FOLDER, LOG_DB_FILE = os.path.split(LOG_DB_PATH)
CACHE_DB_PATH = configpaths.get('CACHE_DB') CACHE_DB_PATH = configpaths.get('CACHE_DB')
import logging LOGS_SQL_STATEMENT = '''
log = logging.getLogger('gajim.c.logger')
class Logger:
def __init__(self):
self._jid_ids = {}
self.con = None
self.commit_timout_id = None
if os.path.isdir(LOG_DB_PATH):
print(_('%s is a directory but should be a file') % LOG_DB_PATH)
print(_('Gajim will now exit'))
sys.exit()
if os.path.isdir(CACHE_DB_PATH):
print(_('%s or %s is a directory but should be a file') % CACHE_DB_PATH)
print(_('Gajim will now exit'))
sys.exit()
if not os.path.exists(LOG_DB_PATH):
if os.path.exists(CACHE_DB_PATH):
os.remove(CACHE_DB_PATH)
self.create_log_db()
self.init_vars()
if not os.path.exists(CACHE_DB_PATH):
self.create_cache_db()
self.attach_cache_database()
def create_log_db(self):
print(_('creating logs database'))
con = sqlite.connect(LOG_DB_PATH)
os.chmod(LOG_DB_PATH, 0o600) # rw only for us
cur = con.cursor()
# create the tables
# kind can be
# status, gcstatus, gc_msg, (we only recv for those 3),
# single_msg_recv, chat_msg_recv, chat_msg_sent, single_msg_sent
# to meet all our needs
# logs.jid_id --> jids.jid_id but Sqlite doesn't do FK etc so it's done in python code
# jids.jid text column will be JID if TC-related, room_jid if GC-related,
# ROOM_JID/nick if pm-related.
# also check optparser.py, which updates databases on gajim updates
cur.executescript(
'''
CREATE TABLE jids( CREATE TABLE jids(
jid_id INTEGER PRIMARY KEY AUTOINCREMENT UNIQUE, jid_id INTEGER PRIMARY KEY AUTOINCREMENT UNIQUE,
jid TEXT UNIQUE, jid TEXT UNIQUE,
type INTEGER type INTEGER
); );
CREATE TABLE unread_messages( CREATE TABLE unread_messages(
message_id INTEGER PRIMARY KEY AUTOINCREMENT UNIQUE, message_id INTEGER PRIMARY KEY AUTOINCREMENT UNIQUE,
jid_id INTEGER, jid_id INTEGER,
shown BOOLEAN default 0 shown BOOLEAN default 0
); );
CREATE INDEX idx_unread_messages_jid_id ON unread_messages (jid_id); CREATE INDEX idx_unread_messages_jid_id ON unread_messages (jid_id);
CREATE TABLE logs( CREATE TABLE logs(
log_line_id INTEGER PRIMARY KEY AUTOINCREMENT UNIQUE, log_line_id INTEGER PRIMARY KEY AUTOINCREMENT UNIQUE,
account_id INTEGER, account_id INTEGER,
@ -129,48 +81,32 @@ class Logger:
encryption_state TEXT, encryption_state TEXT,
marker INTEGER marker INTEGER
); );
CREATE TABLE last_archive_message( CREATE TABLE last_archive_message(
jid_id INTEGER PRIMARY KEY UNIQUE, jid_id INTEGER PRIMARY KEY UNIQUE,
last_mam_id TEXT, last_mam_id TEXT,
oldest_mam_timestamp TEXT, oldest_mam_timestamp TEXT,
last_muc_timestamp TEXT last_muc_timestamp TEXT
); );
CREATE INDEX idx_logs_jid_id_time ON logs (jid_id, time DESC); CREATE INDEX idx_logs_jid_id_time ON logs (jid_id, time DESC);
CREATE INDEX idx_logs_stanza_id ON logs (stanza_id); CREATE INDEX idx_logs_stanza_id ON logs (stanza_id);
PRAGMA user_version=1;
''' '''
)
con.commit() CACHE_SQL_STATEMENT = '''
con.close()
def create_cache_db(self):
print(_('creating cache database'))
con = sqlite.connect(CACHE_DB_PATH)
os.chmod(CACHE_DB_PATH, 0o600) # rw only for us
cur = con.cursor()
cur.executescript(
'''
CREATE TABLE transports_cache ( CREATE TABLE transports_cache (
transport TEXT UNIQUE, transport TEXT UNIQUE,
type INTEGER type INTEGER
); );
CREATE TABLE caps_cache ( CREATE TABLE caps_cache (
hash_method TEXT, hash_method TEXT,
hash TEXT, hash TEXT,
data BLOB, data BLOB,
last_seen INTEGER); last_seen INTEGER);
CREATE TABLE rooms_last_message_time( CREATE TABLE rooms_last_message_time(
jid_id INTEGER PRIMARY KEY UNIQUE, jid_id INTEGER PRIMARY KEY UNIQUE,
time INTEGER time INTEGER
); );
CREATE TABLE roster_entry(
CREATE TABLE IF NOT EXISTS roster_entry(
account_jid_id INTEGER, account_jid_id INTEGER,
jid_id INTEGER, jid_id INTEGER,
name TEXT, name TEXT,
@ -179,19 +115,139 @@ class Logger:
avatar_sha TEXT, avatar_sha TEXT,
PRIMARY KEY (account_jid_id, jid_id) PRIMARY KEY (account_jid_id, jid_id)
); );
CREATE TABLE roster_group(
CREATE TABLE IF NOT EXISTS roster_group(
account_jid_id INTEGER, account_jid_id INTEGER,
jid_id INTEGER, jid_id INTEGER,
group_name TEXT, group_name TEXT,
PRIMARY KEY (account_jid_id, jid_id, group_name) PRIMARY KEY (account_jid_id, jid_id, group_name)
); );
PRAGMA user_version=1;
''' '''
)
log = logging.getLogger('gajim.c.logger')
class Logger:
def __init__(self):
self._jid_ids = {}
self._con = None
self._commit_timout_id = None
self._create_databases()
self._migrate_databases()
self._connect_databases()
self._get_jid_ids_from_db()
def _create_databases(self):
if os.path.isdir(LOG_DB_PATH):
log.error(_('%s is a directory but should be a file'),
LOG_DB_PATH)
sys.exit()
if os.path.isdir(CACHE_DB_PATH):
log.error(_('%s is a directory but should be a file'),
CACHE_DB_PATH)
sys.exit()
if not os.path.exists(LOG_DB_PATH):
if os.path.exists(CACHE_DB_PATH):
os.remove(CACHE_DB_PATH)
self._create(LOGS_SQL_STATEMENT, LOG_DB_PATH)
if not os.path.exists(CACHE_DB_PATH):
self._create(CACHE_SQL_STATEMENT, CACHE_DB_PATH)
@staticmethod
def _create(statement, path):
log.info(_('Creating %s'), path)
con = sqlite.connect(path)
os.chmod(path, 0o600)
try:
con.executescript(statement)
except Exception as error:
log.exception('Error')
con.close()
os.remove(path)
sys.exit()
con.commit() con.commit()
con.close() con.close()
@staticmethod
def _get_user_version(con) -> int:
""" Return the value of PRAGMA user_version. """
return con.execute('PRAGMA user_version').fetchone()[0]
def _migrate_databases(self):
try:
con = sqlite.connect(LOG_DB_PATH)
self._migrate_logs(con)
con.close()
con = sqlite.connect(CACHE_DB_PATH)
self._migrate_cache(con)
con.close()
except Exception:
log.exception('Error')
sys.exit()
def _migrate_logs(self, con):
if self._get_user_version(con) == 0:
# All migrations from 0.16.9 until 1.0.0
statements = [
'ALTER TABLE logs ADD COLUMN "account_id" INTEGER',
'ALTER TABLE logs ADD COLUMN "stanza_id" TEXT',
'ALTER TABLE logs ADD COLUMN "encryption" TEXT',
'ALTER TABLE logs ADD COLUMN "encryption_state" TEXT',
'ALTER TABLE logs ADD COLUMN "marker" INTEGER',
'ALTER TABLE logs ADD COLUMN "additional_data" TEXT',
'''CREATE TABLE IF NOT EXISTS last_archive_message(
jid_id INTEGER PRIMARY KEY UNIQUE,
last_mam_id TEXT,
oldest_mam_timestamp TEXT,
last_muc_timestamp TEXT
)''',
'''CREATE INDEX IF NOT EXISTS idx_logs_stanza_id
ON logs(stanza_id)''',
'PRAGMA user_version=1'
]
self._execute_multiple(con, statements)
if self._get_user_version(con) < 2:
pass
def _migrate_cache(self, con):
if self._get_user_version(con) == 0:
# All migrations from 0.16.9 until 1.0.0
statements = [
'ALTER TABLE roster_entry ADD COLUMN "avatar_sha" TEXT',
'PRAGMA user_version=1'
]
self._execute_multiple(con, statements)
if self._get_user_version(con) < 2:
pass
@staticmethod
def _execute_multiple(con, statements):
"""
Execute mutliple statements with the option to fail on duplicates
but still continue
"""
for sql in statements:
try:
con.execute(sql)
con.commit()
except sqlite.OperationalError as error:
if str(error).startswith('duplicate column name:'):
log.info(error)
else:
log.exception('Error')
sys.exit()
@staticmethod @staticmethod
def namedtuple_factory(cursor, row): def namedtuple_factory(cursor, row):
""" """
@ -209,55 +265,33 @@ class Logger:
def dispatch(self, event, error): def dispatch(self, event, error):
app.ged.raise_event(event, None, str(error)) app.ged.raise_event(event, None, str(error))
def close_db(self): def _connect_databases(self):
if self.con: self._con = sqlite.connect(
self.con.close() LOG_DB_PATH, timeout=20.0, isolation_level='IMMEDIATE')
self.con = None
self.cur = None
def open_db(self): self._con.row_factory = self.namedtuple_factory
self.close_db()
# FIXME: sqlite3_open wants UTF8 strings. So a path with
# non-ascii chars doesn't work. See #2812 and
# http://lists.initd.org/pipermail/pysqlite/2005-August/000134.html
back = os.getcwd()
os.chdir(LOG_DB_FOLDER)
# if locked, wait up to 20 sec to unlock
# before raise (hopefully should be enough)
self.con = sqlite.connect(LOG_DB_FILE, timeout=20.0,
isolation_level='IMMEDIATE')
os.chdir(back)
self.con.row_factory = self.namedtuple_factory
# DB functions # DB functions
self.con.create_function("like", 1, self._like) self._con.create_function("like", 1, self._like)
self.con.create_function("get_timeout", 0, self._get_timeout) self._con.create_function("get_timeout", 0, self._get_timeout)
self.cur = self.con.cursor() self._set_synchronous(False)
self.set_synchronous(False)
def attach_cache_database(self):
try: try:
self.cur.execute("ATTACH DATABASE '%s' AS cache" % \ self._con.execute("ATTACH DATABASE '%s' AS cache" %
CACHE_DB_PATH.replace("'", "''")) CACHE_DB_PATH.replace("'", "''"))
except sqlite.Error as e: except Exception as error:
log.debug("Failed to attach cache database: %s" % str(e)) log.exception('Error')
self._con.close()
sys.exit()
def set_synchronous(self, sync): def _set_synchronous(self, sync):
try: try:
if sync: if sync:
self.cur.execute("PRAGMA synchronous = NORMAL") self._con.execute("PRAGMA synchronous = NORMAL")
else: else:
self.cur.execute("PRAGMA synchronous = OFF") self._con.execute("PRAGMA synchronous = OFF")
except sqlite.Error as e: except sqlite.Error as e:
log.debug("Failed to set_synchronous(%s): %s" % (sync, str(e))) log.exception('Error')
def init_vars(self):
self.open_db()
self.get_jid_ids_from_db()
@staticmethod @staticmethod
def _get_timeout(): def _get_timeout():
@ -277,29 +311,29 @@ class Logger:
def commit(self): def commit(self):
try: try:
self.con.commit() self._con.commit()
except sqlite.OperationalError as e: except sqlite.OperationalError as e:
print(str(e), file=sys.stderr) print(str(e), file=sys.stderr)
self.commit_timout_id = None self._commit_timout_id = None
return False return False
def _timeout_commit(self): def _timeout_commit(self):
if self.commit_timout_id: if self._commit_timout_id:
return return
self.commit_timout_id = GLib.timeout_add(500, self.commit) self._commit_timout_id = GLib.timeout_add(500, self.commit)
def simple_commit(self, sql_to_commit): def simple_commit(self, sql_to_commit):
""" """
Helper to commit Helper to commit
""" """
self.cur.execute(sql_to_commit) self._con.execute(sql_to_commit)
self._timeout_commit() self._timeout_commit()
def get_jid_ids_from_db(self): def _get_jid_ids_from_db(self):
""" """
Load all jid/jid_id tuples into a dict for faster access Load all jid/jid_id tuples into a dict for faster access
""" """
rows = self.con.execute( rows = self._con.execute(
'SELECT jid_id, jid, type FROM jids').fetchall() 'SELECT jid_id, jid, type FROM jids').fetchall()
for row in rows: for row in rows:
self._jid_ids[row.jid] = row self._jid_ids[row.jid] = row
@ -327,8 +361,8 @@ class Logger:
""" """
Return True if it's a room jid, False if it's not, None if we don't know Return True if it's a room jid, False if it's not, None if we don't know
""" """
self.cur.execute('SELECT type FROM jids WHERE jid=?', (jid,)) row = self._con.execute(
row = self.cur.fetchone() 'SELECT type FROM jids WHERE jid=?', (jid,)).fetchone()
if row is None: if row is None:
return None return None
else: else:
@ -380,7 +414,7 @@ class Logger:
return result.jid_id return result.jid_id
sql = 'SELECT jid_id, jid, type FROM jids WHERE jid = ?' sql = 'SELECT jid_id, jid, type FROM jids WHERE jid = ?'
row = self.con.execute(sql, [jid]).fetchone() row = self._con.execute(sql, [jid]).fetchone()
if row is not None: if row is not None:
self._jid_ids[jid] = row self._jid_ids[jid] = row
return row.jid_id return row.jid_id
@ -390,7 +424,7 @@ class Logger:
'Unable to insert new JID because type is missing') 'Unable to insert new JID because type is missing')
sql = 'INSERT INTO jids (jid, type) VALUES (?, ?)' sql = 'INSERT INTO jids (jid, type) VALUES (?, ?)'
lastrowid = self.con.execute(sql, (jid, type_)).lastrowid lastrowid = self._con.execute(sql, (jid, type_)).lastrowid
Row = namedtuple('Row', 'jid_id jid type') Row = namedtuple('Row', 'jid_id jid type')
self._jid_ids[jid] = Row(lastrowid, jid, type_) self._jid_ids[jid] = Row(lastrowid, jid, type_)
self._timeout_commit() self._timeout_commit()
@ -576,9 +610,8 @@ class Logger:
""" """
all_messages = [] all_messages = []
try: try:
self.cur.execute( unread_results = self._con.execute(
'SELECT message_id, shown from unread_messages') 'SELECT message_id, shown from unread_messages').fetchall()
unread_results = self.cur.fetchall()
except Exception: except Exception:
unread_results = [] unread_results = []
for message in unread_results: for message in unread_results:
@ -587,20 +620,19 @@ class Logger:
# here we get infos for that message, and related jid from jids table # here we get infos for that message, and related jid from jids table
# do NOT change order of SELECTed things, unless you change function(s) # do NOT change order of SELECTed things, unless you change function(s)
# that called this function # that called this function
self.cur.execute(''' result = self._con.execute('''
SELECT logs.log_line_id, logs.message, logs.time, logs.subject, SELECT logs.log_line_id, logs.message, logs.time, logs.subject,
jids.jid, logs.additional_data jids.jid, logs.additional_data
FROM logs, jids FROM logs, jids
WHERE logs.log_line_id = %d AND logs.jid_id = jids.jid_id WHERE logs.log_line_id = %d AND logs.jid_id = jids.jid_id
''' % msg_log_id ''' % msg_log_id
) ).fetchone()
results = self.cur.fetchone() if result is None:
if results is None:
# Log line is no more in logs table. remove it from unread_messages # Log line is no more in logs table. remove it from unread_messages
self.set_read_messages([msg_log_id]) self.set_read_messages([msg_log_id])
continue continue
all_messages.append((results, shown)) all_messages.append((result, shown))
return all_messages return all_messages
def get_last_conversation_lines(self, account, jid, pending): def get_last_conversation_lines(self, account, jid, pending):
@ -644,7 +676,7 @@ class Logger:
kinds=', '.join(kinds)) kinds=', '.join(kinds))
try: try:
messages = self.con.execute( messages = self._con.execute(
sql, tuple(jids) + (restore, pending)).fetchall() sql, tuple(jids) + (restore, pending)).fetchall()
except sqlite.DatabaseError: except sqlite.DatabaseError:
self.dispatch('DB_ERROR', self.dispatch('DB_ERROR',
@ -691,7 +723,7 @@ class Logger:
ORDER BY time, log_line_id ORDER BY time, log_line_id
'''.format(jids=', '.join('?' * len(jids))) '''.format(jids=', '.join('?' * len(jids)))
return self.con.execute(sql, tuple(jids) + return self._con.execute(sql, tuple(jids) +
(date.timestamp(), (date.timestamp(),
(date + delta).timestamp())).fetchall() (date + delta).timestamp())).fetchall()
@ -734,7 +766,7 @@ class Logger:
'''.format(jids=', '.join('?' * len(jids)), '''.format(jids=', '.join('?' * len(jids)),
date_search=between if date else '') date_search=between if date else '')
return self.con.execute(sql, tuple(jids) + (query,)).fetchall() return self._con.execute(sql, tuple(jids) + (query,)).fetchall()
def get_days_with_logs(self, account, jid, year, month): def get_days_with_logs(self, account, jid, year, month):
""" """
@ -772,7 +804,7 @@ class Logger:
""".format(jids=', '.join('?' * len(jids)), """.format(jids=', '.join('?' * len(jids)),
kinds=', '.join(kinds)) kinds=', '.join(kinds))
return self.con.execute(sql, tuple(jids) + return self._con.execute(sql, tuple(jids) +
(date.timestamp(), (date.timestamp(),
(date + delta).timestamp())).fetchall() (date + delta).timestamp())).fetchall()
@ -800,7 +832,7 @@ class Logger:
# fetchone() returns always at least one Row with all # fetchone() returns always at least one Row with all
# attributes set to None because of the MAX() function # attributes set to None because of the MAX() function
return self.con.execute(sql, tuple(jids)).fetchone().time return self._con.execute(sql, tuple(jids)).fetchone().time
def get_first_date_that_has_logs(self, account, jid): def get_first_date_that_has_logs(self, account, jid):
""" """
@ -826,7 +858,7 @@ class Logger:
# fetchone() returns always at least one Row with all # fetchone() returns always at least one Row with all
# attributes set to None because of the MIN() function # attributes set to None because of the MIN() function
return self.con.execute(sql, tuple(jids)).fetchone().time return self._con.execute(sql, tuple(jids)).fetchone().time
def get_date_has_logs(self, account, jid, date): def get_date_has_logs(self, account, jid, date):
""" """
@ -856,7 +888,7 @@ class Logger:
AND time BETWEEN ? AND ? AND time BETWEEN ? AND ?
'''.format(jids=', '.join('?' * len(jids))) '''.format(jids=', '.join('?' * len(jids)))
return self.con.execute(sql, tuple(jids) + return self._con.execute(sql, tuple(jids) +
(date.timestamp(), (date.timestamp(),
(date + delta).timestamp())).fetchone() (date + delta).timestamp())).fetchone()
@ -875,7 +907,7 @@ class Logger:
NATURAL JOIN jids WHERE jid = ? NATURAL JOIN jids WHERE jid = ?
''' '''
row = self.con.execute(sql, (jid,)).fetchone() row = self._con.execute(sql, (jid,)).fetchone()
if not row: if not row:
return self.get_last_date_that_has_logs(account, jid) return self.get_last_date_that_has_logs(account, jid)
return row.time return row.time
@ -896,7 +928,7 @@ class Logger:
(SELECT time FROM rooms_last_message_time (SELECT time FROM rooms_last_message_time
WHERE jid_id = :jid_id AND time >= :time), :time))''' WHERE jid_id = :jid_id AND time >= :time), :time))'''
self.con.execute(sql, {"jid_id": jid_id, "time": timestamp}) self._con.execute(sql, {"jid_id": jid_id, "time": timestamp})
self._timeout_commit() self._timeout_commit()
def save_transport_type(self, jid, type_): def save_transport_type(self, jid, type_):
@ -907,11 +939,10 @@ class Logger:
if not type_id: if not type_id:
# unknown type # unknown type
return return
self.cur.execute( result = self._con.execute(
'SELECT type from transports_cache WHERE transport = "%s"' % jid) 'SELECT type from transports_cache WHERE transport = "%s"' % jid).fetchone()
results = self.cur.fetchone() if result:
if results: if result.type == type_id:
if results.type == type_id:
return return
sql = 'UPDATE transports_cache SET type = %d WHERE transport = "%s"' %\ sql = 'UPDATE transports_cache SET type = %d WHERE transport = "%s"' %\
(type_id, jid) (type_id, jid)
@ -924,9 +955,7 @@ class Logger:
""" """
Return all the type of the transports in DB Return all the type of the transports in DB
""" """
self.cur.execute( results = self._con.execute('SELECT * from transports_cache').fetchall()
'SELECT * from transports_cache')
results = self.cur.fetchall()
if not results: if not results:
return {} return {}
answer = {} answer = {}
@ -954,7 +983,7 @@ class Logger:
# the data field contains binary object (gzipped data), this is a hack # the data field contains binary object (gzipped data), this is a hack
# to get that data without trying to convert it to unicode # to get that data without trying to convert it to unicode
try: try:
self.cur.execute('SELECT hash_method, hash, data FROM caps_cache;') rows = self._con.execute('SELECT hash_method, hash, data FROM caps_cache;')
except sqlite.OperationalError: except sqlite.OperationalError:
# might happen when there's no caps_cache table yet # might happen when there's no caps_cache table yet
# -- there's no data to read anyway then # -- there's no data to read anyway then
@ -962,7 +991,7 @@ class Logger:
# list of corrupted entries that will be removed # list of corrupted entries that will be removed
to_be_removed = [] to_be_removed = []
for row in self.cur: for row in rows:
# for each row: unpack the data field # for each row: unpack the data field
# (format: (category, type, name, category, type, name, ... # (format: (category, type, name, category, type, name, ...
# ..., 'FEAT', feature1, feature2, ...).join(' ')) # ..., 'FEAT', feature1, feature2, ...).join(' '))
@ -1013,7 +1042,7 @@ class Logger:
gzip.write(data.encode('utf-8')) gzip.write(data.encode('utf-8'))
gzip.close() gzip.close()
data = string.getvalue() data = string.getvalue()
self.cur.execute(''' self._con.execute('''
INSERT INTO caps_cache ( hash_method, hash, data, last_seen ) INSERT INTO caps_cache ( hash_method, hash, data, last_seen )
VALUES (?, ?, ?, ?); VALUES (?, ?, ?, ?);
''', (hash_method, hash_, memoryview(data), int(time.time()))) ''', (hash_method, hash_, memoryview(data), int(time.time())))
@ -1076,10 +1105,10 @@ class Logger:
jid_id = self.get_jid_id(jid) jid_id = self.get_jid_id(jid)
except exceptions.PysqliteOperationalError as e: except exceptions.PysqliteOperationalError as e:
raise exceptions.PysqliteOperationalError(str(e)) raise exceptions.PysqliteOperationalError(str(e))
self.cur.execute( self._con.execute(
'DELETE FROM roster_group WHERE account_jid_id=? AND jid_id=?', 'DELETE FROM roster_group WHERE account_jid_id=? AND jid_id=?',
(account_jid_id, jid_id)) (account_jid_id, jid_id))
self.cur.execute( self._con.execute(
'DELETE FROM roster_entry WHERE account_jid_id=? AND jid_id=?', 'DELETE FROM roster_entry WHERE account_jid_id=? AND jid_id=?',
(account_jid_id, jid_id)) (account_jid_id, jid_id))
self._timeout_commit() self._timeout_commit()
@ -1101,18 +1130,18 @@ class Logger:
# Update groups information # Update groups information
# First we delete all previous groups information # First we delete all previous groups information
self.cur.execute( self._con.execute(
'DELETE FROM roster_group WHERE account_jid_id=? AND jid_id=?', 'DELETE FROM roster_group WHERE account_jid_id=? AND jid_id=?',
(account_jid_id, jid_id)) (account_jid_id, jid_id))
# Then we add all new groups information # Then we add all new groups information
for group in groups: for group in groups:
self.cur.execute('INSERT INTO roster_group VALUES(?, ?, ?)', self._con.execute('INSERT INTO roster_group VALUES(?, ?, ?)',
(account_jid_id, jid_id, group)) (account_jid_id, jid_id, group))
if name is None: if name is None:
name = '' name = ''
self.cur.execute(''' self._con.execute('''
REPLACE INTO roster_entry REPLACE INTO roster_entry
(account_jid_id, jid_id, name, subscription, ask) (account_jid_id, jid_id, name, subscription, ask)
VALUES(?, ?, ?, ?, ?)''', ( VALUES(?, ?, ?, ?, ?)''', (
@ -1130,11 +1159,11 @@ class Logger:
account_jid_id = self.get_jid_id(account_jid, type_=JIDConstant.NORMAL_TYPE) account_jid_id = self.get_jid_id(account_jid, type_=JIDConstant.NORMAL_TYPE)
# First we fill data with roster_entry informations # First we fill data with roster_entry informations
self.cur.execute(''' rows = self._con.execute('''
SELECT j.jid, re.jid_id, re.name, re.subscription, re.ask, re.avatar_sha SELECT j.jid, re.jid_id, re.name, re.subscription, re.ask, re.avatar_sha
FROM roster_entry re, jids j FROM roster_entry re, jids j
WHERE re.account_jid_id=? AND j.jid_id=re.jid_id''', (account_jid_id,)) WHERE re.account_jid_id=? AND j.jid_id=re.jid_id''', (account_jid_id,))
for row in self.cur: for row in rows:
#jid, jid_id, name, subscription, ask #jid, jid_id, name, subscription, ask
jid = row.jid jid = row.jid
name = row.name name = row.name
@ -1157,11 +1186,11 @@ class Logger:
# Then we add group for roster entries # Then we add group for roster entries
for jid in data: for jid in data:
self.cur.execute(''' rows = self._con.execute('''
SELECT group_name FROM roster_group SELECT group_name FROM roster_group
WHERE account_jid_id=? AND jid_id=?''', WHERE account_jid_id=? AND jid_id=?''',
(account_jid_id, data[jid]['id'])) (account_jid_id, data[jid]['id']))
for row in self.cur: for row in rows:
group_name = row.group_name group_name = row.group_name
data[jid]['groups'].append(group_name) data[jid]['groups'].append(group_name)
del data[jid]['id'] del data[jid]['id']
@ -1186,7 +1215,7 @@ class Logger:
DELETE FROM roster_group WHERE account_jid_id = {jid_id}; DELETE FROM roster_group WHERE account_jid_id = {jid_id};
'''.format(jid_id=jid_id) '''.format(jid_id=jid_id)
self.con.executescript(sql) self._con.executescript(sql)
self._timeout_commit() self._timeout_commit()
def search_for_duplicate(self, account, jid, timestamp, msg): def search_for_duplicate(self, account, jid, timestamp, msg):
@ -1216,7 +1245,7 @@ class Logger:
AND time BETWEEN ? AND ? AND time BETWEEN ? AND ?
''' '''
result = self.con.execute( result = self._con.execute(
sql, (jid, msg, account_id, start_time, end_time)).fetchone() sql, (jid, msg, account_id, start_time, end_time)).fetchone()
if result is not None: if result is not None:
@ -1263,14 +1292,14 @@ class Logger:
WHERE stanza_id IN ({values}) WHERE stanza_id IN ({values})
AND jid_id = ? AND account_id = ? LIMIT 1 AND jid_id = ? AND account_id = ? LIMIT 1
'''.format(values=', '.join('?' * len(ids))) '''.format(values=', '.join('?' * len(ids)))
result = self.con.execute( result = self._con.execute(
sql, tuple(ids) + (archive_id, account_id)).fetchone() sql, tuple(ids) + (archive_id, account_id)).fetchone()
else: else:
sql = ''' sql = '''
SELECT stanza_id FROM logs SELECT stanza_id FROM logs
WHERE stanza_id IN ({values}) AND account_id = ? AND kind != ? LIMIT 1 WHERE stanza_id IN ({values}) AND account_id = ? AND kind != ? LIMIT 1
'''.format(values=', '.join('?' * len(ids))) '''.format(values=', '.join('?' * len(ids)))
result = self.con.execute( result = self._con.execute(
sql, tuple(ids) + (account_id, KindConstant.GC_MSG)).fetchone() sql, tuple(ids) + (account_id, KindConstant.GC_MSG)).fetchone()
if result is not None: if result is not None:
@ -1324,7 +1353,7 @@ class Logger:
'''.format(columns=', '.join(kwargs.keys()), '''.format(columns=', '.join(kwargs.keys()),
values=', '.join('?' * len(kwargs))) values=', '.join('?' * len(kwargs)))
lastrowid = self.con.execute( lastrowid = self._con.execute(
sql, (account_id, jid_id, time_, kind) + tuple(kwargs.values())).lastrowid sql, (account_id, jid_id, time_, kind) + tuple(kwargs.values())).lastrowid
log.info('Insert into DB: jid: %s, time: %s, kind: %s, stanza_id: %s', log.info('Insert into DB: jid: %s, time: %s, kind: %s, stanza_id: %s',
@ -1333,7 +1362,7 @@ class Logger:
if unread and kind == KindConstant.CHAT_MSG_RECV: if unread and kind == KindConstant.CHAT_MSG_RECV:
sql = '''INSERT INTO unread_messages (message_id, jid_id) sql = '''INSERT INTO unread_messages (message_id, jid_id)
VALUES (?, (SELECT jid_id FROM jids WHERE jid = ?))''' VALUES (?, (SELECT jid_id FROM jids WHERE jid = ?))'''
self.con.execute(sql, (lastrowid, jid)) self._con.execute(sql, (lastrowid, jid))
self._timeout_commit() self._timeout_commit()
@ -1358,7 +1387,7 @@ class Logger:
UPDATE roster_entry SET avatar_sha = ? UPDATE roster_entry SET avatar_sha = ?
WHERE account_jid_id = ? AND jid_id = ? WHERE account_jid_id = ? AND jid_id = ?
''' '''
self.con.execute(sql, (sha, account_jid_id, jid_id)) self._con.execute(sql, (sha, account_jid_id, jid_id))
self._timeout_commit() self._timeout_commit()
def get_archive_timestamp(self, jid, type_=None): def get_archive_timestamp(self, jid, type_=None):
@ -1370,7 +1399,7 @@ class Logger:
""" """
jid_id = self.get_jid_id(jid, type_=type_) jid_id = self.get_jid_id(jid, type_=type_)
sql = '''SELECT * FROM last_archive_message WHERE jid_id = ?''' sql = '''SELECT * FROM last_archive_message WHERE jid_id = ?'''
return self.con.execute(sql, (jid_id,)).fetchone() return self._con.execute(sql, (jid_id,)).fetchone()
def set_archive_timestamp(self, jid, **kwargs): def set_archive_timestamp(self, jid, **kwargs):
""" """
@ -1391,7 +1420,7 @@ class Logger:
exists = self.get_archive_timestamp(jid) exists = self.get_archive_timestamp(jid)
if not exists: if not exists:
sql = '''INSERT INTO last_archive_message VALUES (?, ?, ?, ?)''' sql = '''INSERT INTO last_archive_message VALUES (?, ?, ?, ?)'''
self.con.execute(sql, (jid_id, self._con.execute(sql, (jid_id,
kwargs.get('last_mam_id', None), kwargs.get('last_mam_id', None),
kwargs.get('oldest_mam_timestamp', None), kwargs.get('oldest_mam_timestamp', None),
kwargs.get('last_muc_timestamp', None))) kwargs.get('last_muc_timestamp', None)))
@ -1399,6 +1428,6 @@ class Logger:
args = ' = ?, '.join(kwargs.keys()) + ' = ?' args = ' = ?, '.join(kwargs.keys()) + ' = ?'
sql = '''UPDATE last_archive_message SET {} sql = '''UPDATE last_archive_message SET {}
WHERE jid_id = ?'''.format(args) WHERE jid_id = ?'''.format(args)
self.con.execute(sql, tuple(kwargs.values()) + (jid_id,)) self._con.execute(sql, tuple(kwargs.values()) + (jid_id,))
log.info('Save archive timestamps: %s', kwargs) log.info('Save archive timestamps: %s', kwargs)
self._timeout_commit() self._timeout_commit()

View File

@ -1,5 +1,5 @@
# -*- coding:utf-8 -*- # -*- coding:utf-8 -*-
## src/common/optparser.py ## gajim/common/optparser.py
## ##
## Copyright (C) 2003-2005 Vincent Hanquez <tab AT snarc.org> ## Copyright (C) 2003-2005 Vincent Hanquez <tab AT snarc.org>
## Copyright (C) 2003-2014 Yann Leboulanger <asterix AT lagaule.org> ## Copyright (C) 2003-2014 Yann Leboulanger <asterix AT lagaule.org>
@ -10,6 +10,7 @@
## Brendan Taylor <whateley AT gmail.com> ## Brendan Taylor <whateley AT gmail.com>
## Tomasz Melcer <liori AT exroot.org> ## Tomasz Melcer <liori AT exroot.org>
## Stephan Erb <steve-e AT h3c.de> ## Stephan Erb <steve-e AT h3c.de>
## Copyright (C) 2018 Philipp Hörist <philipp AT hoerist.com>
## ##
## This file is part of Gajim. ## This file is part of Gajim.
## ##
@ -29,17 +30,15 @@
import os import os
import sys import sys
import re import re
from time import time import logging
from gajim.common import app from gajim.common import app
from gajim.common import helpers
from gajim.common import caps_cache from gajim.common import caps_cache
import sqlite3 as sqlite
from gajim.common import logger
import logging
log = logging.getLogger('gajim.c.optparser') log = logging.getLogger('gajim.c.optparser')
class OptionsParser: class OptionsParser:
def __init__(self, filename): def __init__(self, filename):
self.__filename = os.path.realpath(filename) self.__filename = os.path.realpath(filename)
@ -150,25 +149,13 @@ class OptionsParser:
self.update_config_to_016101() self.update_config_to_016101()
if old < [0, 16, 10, 2] and new >= [0, 16, 10, 2]: if old < [0, 16, 10, 2] and new >= [0, 16, 10, 2]:
self.update_config_to_016102() self.update_config_to_016102()
if old < [0, 16, 10, 3] and new >= [0, 16, 10, 3]:
self.update_config_to_016103()
if old < [0, 16, 10, 4] and new >= [0, 16, 10, 4]: if old < [0, 16, 10, 4] and new >= [0, 16, 10, 4]:
self.update_config_to_016104() self.update_config_to_016104()
if old < [0, 16, 10, 5] and new >= [0, 16, 10, 5]: if old < [0, 16, 10, 5] and new >= [0, 16, 10, 5]:
self.update_config_to_016105() self.update_config_to_016105()
if old < [0, 16, 11, 1] and new >= [0, 16, 11, 1]:
self.update_config_to_016111()
if old < [0, 16, 11, 2] and new >= [0, 16, 11, 2]:
self.update_config_to_016112()
if old < [0, 98, 2] and new >= [0, 98, 2]:
self.update_config_to_0982()
if old < [0, 98, 3] and new >= [0, 98, 3]: if old < [0, 98, 3] and new >= [0, 98, 3]:
self.update_config_to_0983() self.update_config_to_0983()
if old < [0, 99, 2] and new >= [0, 99, 2]:
self.update_config_to_0992()
app.logger.init_vars()
app.logger.attach_cache_database()
app.config.set('version', new_version) app.config.set('version', new_version)
caps_cache.capscache.initialize_from_db() caps_cache.capscache.initialize_from_db()
@ -193,20 +180,6 @@ class OptionsParser:
app.config.set_per('accounts', account, 'file_transfer_proxies', app.config.set_per('accounts', account, 'file_transfer_proxies',
proxies_str) proxies_str)
@staticmethod
def call_sql(db_path, sql):
con = sqlite.connect(db_path)
cur = con.cursor()
try:
cur.executescript(sql)
con.commit()
except sqlite.OperationalError as e:
if str(e).startswith('duplicate column name:'):
log.info(str(e))
else:
log.exception('Error')
con.close()
def update_config_to_01401(self): def update_config_to_01401(self):
if 'autodetect_browser_mailer' not in self.old_values or 'openwith' \ if 'autodetect_browser_mailer' not in self.old_values or 'openwith' \
not in self.old_values or \ not in self.old_values or \
@ -253,22 +226,8 @@ class OptionsParser:
for account in self.old_values['accounts'].keys(): for account in self.old_values['accounts'].keys():
app.config.del_per('accounts', account, 'minimized_gc') app.config.del_per('accounts', account, 'minimized_gc')
self.call_sql(logger.LOG_DB_PATH,
'''ALTER TABLE logs
ADD COLUMN 'additional_data' TEXT;'''
)
app.config.set('version', '0.16.10.2') app.config.set('version', '0.16.10.2')
def update_config_to_016103(self):
self.call_sql(logger.LOG_DB_PATH,
'''ALTER TABLE logs ADD COLUMN 'stanza_id' TEXT;
ALTER TABLE logs ADD COLUMN 'encryption' TEXT;
ALTER TABLE logs ADD COLUMN 'encryption_state' TEXT;
ALTER TABLE logs ADD COLUMN 'marker' INTEGER;
'''
)
app.config.set('version', '0.16.10.3')
def update_config_to_016104(self): def update_config_to_016104(self):
app.config.set('emoticons_theme', 'noto-emoticons') app.config.set('emoticons_theme', 'noto-emoticons')
app.config.set('version', '0.16.10.4') app.config.set('version', '0.16.10.4')
@ -278,37 +237,6 @@ class OptionsParser:
app.config.set('restore_timeout', -1) app.config.set('restore_timeout', -1)
app.config.set('version', '0.16.10.5') app.config.set('version', '0.16.10.5')
def update_config_to_016111(self):
self.call_sql(logger.CACHE_DB_PATH,
'''ALTER TABLE roster_entry ADD COLUMN 'avatar_sha' TEXT;
'''
)
app.config.set('version', '0.16.11.1')
def update_config_to_016112(self):
self.call_sql(logger.LOG_DB_PATH,
'''
CREATE TABLE IF NOT EXISTS last_archive_message(
jid_id INTEGER PRIMARY KEY UNIQUE,
last_mam_id TEXT,
oldest_mam_timestamp TEXT,
last_muc_timestamp TEXT
);
ALTER TABLE logs ADD COLUMN 'account_id' INTEGER;
'''
)
app.config.set('version', '0.16.11.2')
def update_config_to_0982(self):
# This fixes a typo in update_config_to_016112()
self.call_sql(logger.LOG_DB_PATH,
'''
ALTER TABLE logs ADD COLUMN 'account_id' INTEGER;
ALTER TABLE logs ADD COLUMN 'additional_data' TEXT;
'''
)
app.config.set('version', '0.98.2')
def update_config_to_0983(self): def update_config_to_0983(self):
for account in self.old_values['accounts'].keys(): for account in self.old_values['accounts'].keys():
password = self.old_values['accounts'][account]['password'] password = self.old_values['accounts'][account]['password']
@ -317,12 +245,3 @@ class OptionsParser:
elif password == "libsecret:": elif password == "libsecret:":
app.config.set_per('accounts', account, 'password', '') app.config.set_per('accounts', account, 'password', '')
app.config.set('version', '0.98.3') app.config.set('version', '0.98.3')
def update_config_to_0992(self):
self.call_sql(logger.LOG_DB_PATH,
'''
CREATE INDEX IF NOT EXISTS
idx_logs_stanza_id ON logs (stanza_id);
'''
)
app.config.set('version', '0.99.2')