From 3b6a6216359cb178672237bd63b01f9f6d795972 Mon Sep 17 00:00:00 2001 From: Doug Blank Date: Fri, 29 Jan 2016 09:53:51 -0500 Subject: [PATCH] Database selects/iters can order by any field --- gramps/gen/db/base.py | 2 +- gramps/gen/db/generic.py | 14 +++++++-- gramps/plugins/database/dbapi.py | 50 +++++++++++++++++++++++++++++++- 3 files changed, 62 insertions(+), 4 deletions(-) diff --git a/gramps/gen/db/base.py b/gramps/gen/db/base.py index f3349a601..4f1f3de40 100644 --- a/gramps/gen/db/base.py +++ b/gramps/gen/db/base.py @@ -2050,6 +2050,6 @@ class DbWriteBase(DbReadBase): """ values = [] for (field, direction) in order_by: - values.append(obj.get_field(field)) + values.append(obj.get_field(field, self, ignore_errors=True)) return values diff --git a/gramps/gen/db/generic.py b/gramps/gen/db/generic.py index 515dc7a63..ca9dedc7c 100644 --- a/gramps/gen/db/generic.py +++ b/gramps/gen/db/generic.py @@ -33,7 +33,6 @@ import logging import shutil import bisect import ast -from operator import itemgetter import sys #------------------------------------------------------------------------ @@ -1166,9 +1165,20 @@ class DbGeneric(DbWriteBase, DbReadBase, UpdateCallback, Callback): # just use values and handle to keep small: sorted_items.append((self.eval_order_by(order_by, obj), obj.handle)) # next we sort by fields and direction + def getitem(item, pos): + sort_items = item[0] + if isinstance(sort_items[pos], str): + return sort_items[pos] + elif sort_items[pos] is None: + return "" + else: + # FIXME: should do something clever/recurive to + # sort these meaningfully, and return a string: + return str(sort_items[pos]) pos = len(order_by) - 1 for (field, order) in reversed(order_by): # sort the lasts parts first - sorted_items.sort(key=itemgetter(pos), reverse=(order=="DESC")) + sorted_items.sort(key=lambda item: getitem(item, pos), + reverse=(order=="DESC")) pos -= 1 # now we will look them up again: for (order_by_values, handle) in sorted_items: diff --git a/gramps/plugins/database/dbapi.py b/gramps/plugins/database/dbapi.py index 7572a9ec7..33784dfe0 100644 --- a/gramps/plugins/database/dbapi.py +++ b/gramps/plugins/database/dbapi.py @@ -1032,7 +1032,41 @@ class DBAPI(DbGeneric): if row: return self.get_person_from_handle(row[0]) + def iter_items_order_by_python(self, order_by, class_): + """ + This method is for those iter_items with a order_by, but + can't be done with secondary fields. + """ + # first build sort order: + sorted_items = [] + query = "SELECT blob_data FROM %s;" % class_.__name__ + self.dbapi.execute(query) + rows = self.dbapi.fetchall() + for row in rows: + obj = obj_() + obj.unserialize(row[0]) + # just use values and handle to keep small: + sorted_items.append((self.eval_order_by(order_by, obj), obj.handle)) + # next we sort by fields and direction + pos = len(order_by) - 1 + for (field, order) in reversed(order_by): # sort the lasts parts first + sorted_items.sort(key=itemgetter(pos), reverse=(order=="DESC")) + pos -= 1 + # now we will look them up again: + for (order_by_values, handle) in sorted_items: + yield self._tables[obj_.__name__]["handle_func"](handle) + def iter_items(self, order_by, class_): + # check if order_by fields are secondary + # if so, fine + # else, use Python sorts + if order_by: + secondary_fields = class_.get_secondary_fields() + if not self.check_order_by_fields(class_.__name__, order_by, secondary_fields): + for item in super().iter_items(order_by, class_): + yield item + return + ## Continue with dbapi select if order_by is None: query = "SELECT blob_data FROM %s;" % class_.__name__ else: @@ -1655,6 +1689,19 @@ class DBAPI(DbGeneric): else: return ["blob_data"] # nope, we'll have to expand blob to get all fields + def check_order_by_fields(self, table, order_by, secondary_fields): + """ + Check to make sure all order_by fields are defined. If not, then + we need to do the Python-based order. + + secondary_fields are hashed. + """ + if order_by: + for (field, dir) in order_by: + if self._hash_name(table, field) not in secondary_fields: + return False + return True + def check_where_fields(self, table, where, secondary_fields): """ Check to make sure all where fields are defined. If not, then @@ -1700,7 +1747,8 @@ class DBAPI(DbGeneric): secondary_fields = ([self._hash_name(table, field) for (field, ptype) in self._tables[table]["class_func"].get_secondary_fields()] + ["handle"]) # handle is a sql field, but not listed in secondaries - if not self.check_where_fields(table, where, secondary_fields): + if ((not self.check_where_fields(table, where, secondary_fields)) or + (not self.check_order_by_fields(table, order_by, secondary_fields))): return super().select(table, fields, start, limit, where, order_by) fields = hashed_fields start_time = time.time()