gramps/src/plugins/tool/Check.py

1651 lines
75 KiB
Python
Raw Normal View History

2002-10-20 14:25:16 +00:00
#
# Gramps - a GTK+/GNOME based genealogy program
#
* src/TreeViews/_PersonTreeView.py: Use name_displayer. * src/ReportBase/_ReportUtils.py: Use name_displayer. * src/ReportBase/_CommandLineReport.py: Use name_displayer. * src/ReportBase/_BareReportDialog.py: Use name_displayer. * src/PluginUtils/_Tool.py: Use name_displayer. * src/plugins/TimeLine.py: Use name_displayer. * src/plugins/RelCalc.py: Use name_displayer. * src/plugins/ReadGrdb.py: Use name_displayer. * src/plugins/NarrativeWeb.py: Use name_displayer. * src/plugins/IndivComplete.py: Use name_displayer. * src/plugins/GraphViz.py: Use name_displayer. * src/plugins/FindDupes.py: Use name_displayer. * src/plugins/FamilyGroup.py: Use name_displayer. * src/plugins/DetDescendantReport.py: Use name_displayer. * src/plugins/DetAncestralReport.py: Use name_displayer. * src/plugins/DesGraph.py: Use name_displayer. * src/plugins/DescendReport.py: Use name_displayer. * src/plugins/DescendChart.py: Use name_displayer. * src/plugins/Check.py: Use name_displayer. * src/plugins/Ancestors.py: Use name_displayer. * src/plugins/AncestorReport.py: Use name_displayer. * src/plugins/AncestorChart2.py: Use name_displayer. * src/ObjectSelector/_PersonTreeFrame.py: Use name_displayer. * src/ObjectSelector/_PersonFrame.py: Use name_displayer. * src/Merge/_MergePerson.py: Use name_displayer. * src/GrampsDbUtils/_WriteGedcom.py: Use name_displayer. * src/GrampsDbUtils/_ReadXML.py: Use name_displayer. * src/GrampsDbUtils/_GedcomParse.py: Use name_displayer. * src/FilterEditor/_ShowResults.py: Use name_displayer. * src/FilterEditor/_EditRule.py: Use name_displayer. * src/Editors/_EditPrimary.py: Use name_displayer. * src/Editors/_EditPersonRef.py: Use name_displayer. * src/Editors/_EditPerson.py: Use name_displayer. * src/Editors/_EditName.py: Use name_displayer. * src/Editors/_EditLdsOrd.py: Use name_displayer. * src/Editors/_EditFamily.py: Use name_displayer. * src/DisplayTabs/_PersonRefModel.py: Use name_displayer. * src/DisplayTabs/_NameModel.py: Use name_displayer. * src/DisplayTabs/_ChildModel.py: Use name_displayer. * src/DisplayTabs/_BackRefModel.py: Use name_displayer. * src/DisplayModels/_PeopleModel.py: Use name_displayer. * src/DisplayModels/_FamilyModel.py: Use name_displayer. * src/DataViews/_PersonView.py: Use name_displayer. * src/DataViews/_RelationView.py: Use name_displayer. * src/DataViews/_PedigreeView.py: Use name_displayer. * src/Utils.py: Use name_displayer. * src/SubstKeywords.py: Use name_displayer. * src/Sort.py: Use name_displayer. * src/Reorder.py: Use name_displayer. * src/PageView.py (BookMarkView.add_bookmark): Use name_displayer. * src/Navigation.py: Use name_displayer. * src/DisplayState.py: Use name_displayer. * src/GrampsCfg.py: Use name_displayer. * src/Bookmarks.py (Bookmarks.make_label): Use name_displayer. * src/GrampsDb/Makefile.am (pkgdata_PYTHON): Ship new files. * src/Makefile.am (gdir_PYTHON): Ship ProgressDialog.py svn: r8680
2007-06-28 05:41:40 +00:00
# Copyright (C) 2000-2007 Donald N. Allingham
# Copyright (C) 2008 Brian G. Matherly
2002-10-20 14:25:16 +00:00
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
#
# $Id$
"""Tools/Database Repair/Check and Repair Database"""
2002-10-20 14:25:16 +00:00
2003-06-16 04:57:25 +00:00
#-------------------------------------------------------------------------
#
# python modules
#
#-------------------------------------------------------------------------
from __future__ import with_statement
2002-10-20 14:25:16 +00:00
import os
import cStringIO
from gettext import gettext as _
2009-01-31 11:29:44 +00:00
from gettext import ngettext
2003-06-16 04:57:25 +00:00
2006-03-05 04:31:24 +00:00
#------------------------------------------------------------------------
#
# Set up logging
#
#------------------------------------------------------------------------
import logging
log = logging.getLogger(".CheckRepair")
2003-06-16 04:57:25 +00:00
#-------------------------------------------------------------------------
#
# gtk modules
#
#-------------------------------------------------------------------------
2002-10-20 14:25:16 +00:00
import gtk
#-------------------------------------------------------------------------
#
# GRAMPS modules
2003-06-16 04:57:25 +00:00
#
#-------------------------------------------------------------------------
import gen.lib
2003-06-16 04:57:25 +00:00
import Utils
from gui.utils import ProgressMeter
#import const
import ManagedWindow
from PluginUtils import Tool
2003-06-16 04:57:25 +00:00
from QuestionDialog import OkDialog, MissingMediaDialog
* src/TreeViews/_PersonTreeView.py: Use name_displayer. * src/ReportBase/_ReportUtils.py: Use name_displayer. * src/ReportBase/_CommandLineReport.py: Use name_displayer. * src/ReportBase/_BareReportDialog.py: Use name_displayer. * src/PluginUtils/_Tool.py: Use name_displayer. * src/plugins/TimeLine.py: Use name_displayer. * src/plugins/RelCalc.py: Use name_displayer. * src/plugins/ReadGrdb.py: Use name_displayer. * src/plugins/NarrativeWeb.py: Use name_displayer. * src/plugins/IndivComplete.py: Use name_displayer. * src/plugins/GraphViz.py: Use name_displayer. * src/plugins/FindDupes.py: Use name_displayer. * src/plugins/FamilyGroup.py: Use name_displayer. * src/plugins/DetDescendantReport.py: Use name_displayer. * src/plugins/DetAncestralReport.py: Use name_displayer. * src/plugins/DesGraph.py: Use name_displayer. * src/plugins/DescendReport.py: Use name_displayer. * src/plugins/DescendChart.py: Use name_displayer. * src/plugins/Check.py: Use name_displayer. * src/plugins/Ancestors.py: Use name_displayer. * src/plugins/AncestorReport.py: Use name_displayer. * src/plugins/AncestorChart2.py: Use name_displayer. * src/ObjectSelector/_PersonTreeFrame.py: Use name_displayer. * src/ObjectSelector/_PersonFrame.py: Use name_displayer. * src/Merge/_MergePerson.py: Use name_displayer. * src/GrampsDbUtils/_WriteGedcom.py: Use name_displayer. * src/GrampsDbUtils/_ReadXML.py: Use name_displayer. * src/GrampsDbUtils/_GedcomParse.py: Use name_displayer. * src/FilterEditor/_ShowResults.py: Use name_displayer. * src/FilterEditor/_EditRule.py: Use name_displayer. * src/Editors/_EditPrimary.py: Use name_displayer. * src/Editors/_EditPersonRef.py: Use name_displayer. * src/Editors/_EditPerson.py: Use name_displayer. * src/Editors/_EditName.py: Use name_displayer. * src/Editors/_EditLdsOrd.py: Use name_displayer. * src/Editors/_EditFamily.py: Use name_displayer. * src/DisplayTabs/_PersonRefModel.py: Use name_displayer. * src/DisplayTabs/_NameModel.py: Use name_displayer. * src/DisplayTabs/_ChildModel.py: Use name_displayer. * src/DisplayTabs/_BackRefModel.py: Use name_displayer. * src/DisplayModels/_PeopleModel.py: Use name_displayer. * src/DisplayModels/_FamilyModel.py: Use name_displayer. * src/DataViews/_PersonView.py: Use name_displayer. * src/DataViews/_RelationView.py: Use name_displayer. * src/DataViews/_PedigreeView.py: Use name_displayer. * src/Utils.py: Use name_displayer. * src/SubstKeywords.py: Use name_displayer. * src/Sort.py: Use name_displayer. * src/Reorder.py: Use name_displayer. * src/PageView.py (BookMarkView.add_bookmark): Use name_displayer. * src/Navigation.py: Use name_displayer. * src/DisplayState.py: Use name_displayer. * src/GrampsCfg.py: Use name_displayer. * src/Bookmarks.py (Bookmarks.make_label): Use name_displayer. * src/GrampsDb/Makefile.am (pkgdata_PYTHON): Ship new files. * src/Makefile.am (gdir_PYTHON): Ship ProgressDialog.py svn: r8680
2007-06-28 05:41:40 +00:00
from BasicUtils import name_displayer as _nd
from glade import Glade
2003-06-16 04:57:25 +00:00
#-------------------------------------------------------------------------
#
# Low Level repair
#
#-------------------------------------------------------------------------
def low_level(db):
"""
This is a low-level repair routine.
It is fixing DB inconsistencies such as duplicates.
Returns a (status, name) tuple.
The boolean status indicates the success of the procedure.
The name indicates the problematic table (empty if status is True).
"""
for the_map in [('Person', db.person_map),
('Family', db.family_map),
('Event', db.event_map),
('Place', db.place_map),
('Source', db.source_map),
('Media', db.media_map),
('Repository', db.repository_map),
('Note', db.note_map)]:
print "Low-level repair: table: %s" % the_map[0]
2009-08-15 14:37:14 +00:00
if _table_low_level(db, the_map[1]):
print "Done."
else:
print "Low-level repair: Problem with table: %s" % the_map[0]
2009-08-15 14:37:14 +00:00
return (False, the_map[0])
return (True, '')
def _table_low_level(db,table):
"""
Low level repair for a given db table.
"""
handle_list = table.keys()
dup_handles = set(
[ handle for handle in handle_list if handle_list.count(handle) > 1 ]
)
if not dup_handles:
2009-08-15 14:37:14 +00:00
print " No duplicates found for this table"
return True
# import gen.db
from gen.db import GrampsDBDirAssocCursor
table_cursor = GrampsDBDirAssocCursor(table)
for handle in dup_handles:
print " Duplicates found for handle: %s" % handle
try:
ret = table_cursor.set(handle)
except:
print " Failed setting initial cursor."
return False
for count in range(handle_list.count(handle)-1):
try:
table_cursor.delete()
2009-08-15 14:37:14 +00:00
print " Successfully deleted duplicate #%d" % (count+1)
except:
2009-08-15 14:37:14 +00:00
print " Failed deleting duplicate."
return False
try:
ret = table_cursor.next_dup()
except:
print " Failed moving the cursor."
return False
table_cursor.close()
table.sync()
return True
2003-06-16 04:57:25 +00:00
#-------------------------------------------------------------------------
2002-10-20 14:25:16 +00:00
#
2003-06-16 04:57:25 +00:00
# runTool
2002-10-20 14:25:16 +00:00
#
#-------------------------------------------------------------------------
class Check(Tool.BatchTool):
def __init__(self, dbstate, uistate, options_class, name, callback=None):
Tool.BatchTool.__init__(self, dbstate, options_class, name)
if self.fail:
return
2005-12-06 06:38:09 +00:00
cli = uistate is None
2005-12-06 06:38:09 +00:00
if self.db.readonly:
2006-03-05 04:31:24 +00:00
# TODO: split plugin in a check and repair part to support
# checking of a read only database
return
# The low-level repair is bypassing the transaction mechanism.
# As such, we run it before starting the transaction.
# We only do this for the dbdir backend.
if self.db.__class__.__name__ == 'GrampsDBDir':
low_level(self.db)
2009-08-15 14:37:14 +00:00
trans = self.db.transaction_begin("", batch=True)
self.db.disable_signals()
checker = CheckIntegrity(dbstate, uistate, trans)
2006-03-05 04:31:24 +00:00
checker.fix_encoding()
checker.check_dates()
2006-03-05 04:31:24 +00:00
checker.cleanup_missing_photos(cli)
checker.cleanup_deleted_name_formats()
2005-12-06 06:38:09 +00:00
2006-03-05 04:31:24 +00:00
prev_total = -1
total = 0
#start with empty objects, broken links can be corrected below then
checker.cleanup_empty_objects()
2006-03-05 04:31:24 +00:00
while prev_total != total:
prev_total = total
2006-03-05 04:31:24 +00:00
checker.check_for_broken_family_links()
checker.check_parent_relationships()
checker.cleanup_empty_families(cli)
checker.cleanup_duplicate_spouses()
total = checker.family_errors()
checker.check_events()
checker.check_person_references()
2006-03-05 04:31:24 +00:00
checker.check_place_references()
checker.check_source_references()
checker.check_media_references()
checker.check_repo_references()
checker.check_note_references()
self.db.transaction_commit(trans, _("Check Integrity"))
self.db.enable_signals()
self.db.request_rebuild()
2006-03-05 04:31:24 +00:00
errs = checker.build_report(cli)
if errs:
2009-08-15 14:37:14 +00:00
Report(uistate, checker.text.getvalue(), cli)
2002-10-20 14:25:16 +00:00
#-------------------------------------------------------------------------
#
#
#
#-------------------------------------------------------------------------
class CheckIntegrity(object):
2002-10-20 14:25:16 +00:00
def __init__(self, dbstate, uistate, trans):
self.db = dbstate.db
self.trans = trans
2002-10-20 14:25:16 +00:00
self.bad_photo = []
self.replaced_photo = []
self.removed_photo = []
2002-10-20 14:25:16 +00:00
self.empty_family = []
self.broken_links = []
self.duplicate_links = []
2002-10-20 14:25:16 +00:00
self.broken_parent_links = []
self.fam_rel = []
self.invalid_events = []
self.invalid_birth_events = []
self.invalid_death_events = []
self.invalid_person_references = []
self.invalid_place_references = []
self.invalid_source_references = []
self.invalid_repo_references = []
self.invalid_media_references = []
self.invalid_note_references = []
self.invalid_dates = []
self.removed_name_format = []
self.empty_objects = {'persons' : [],
'families': [],
'events' : [],
'sources' : [],
'media' : [],
'places' : [],
'repos' : [],
'notes' : [],
}
self.progress = ProgressMeter(_('Checking Database'),'')
2002-10-20 14:25:16 +00:00
def family_errors(self):
2006-04-13 16:16:00 +00:00
return len(self.broken_parent_links) + \
len(self.broken_links) + \
len(self.empty_family) + \
len(self.duplicate_links)
def cleanup_deleted_name_formats(self):
"""
2009-08-15 14:37:14 +00:00
Permanently remove deleted name formats from db.
When user deletes custom name format those are not removed only marked
as "inactive". This method does the cleanup of the name format table,
as well as fixes the display_as, sort_as values for each Name in the db.
"""
self.progress.set_pass(_('Looking for invalid name format references'),
self.db.get_number_of_people())
2009-08-15 14:37:14 +00:00
deleted_name_formats = [number for (number, name, fmt_str,act)
in self.db.name_formats if not act]
# remove the invalid references from all Name objects
for person_handle in self.db.get_person_handles():
person = self.db.get_person_from_handle(person_handle)
p_changed = False
name = person.get_primary_name()
if name.get_sort_as() in deleted_name_formats:
name.set_sort_as(gen.lib.Name.DEF)
p_changed = True
if name.get_display_as() in deleted_name_formats:
name.set_display_as(gen.lib.Name.DEF)
p_changed = True
if p_changed:
person.set_primary_name(name)
a_changed = False
name_list = []
for name in person.get_alternate_names():
if name.get_sort_as() in deleted_name_formats:
name.set_sort_as(gen.lib.Name.DEF)
a_changed = True
if name.get_display_as() in deleted_name_formats:
name.set_display_as(gen.lib.Name.DEF)
a_changed = True
name_list.append(name)
if a_changed:
person.set_alternate_names(name_list)
if p_changed or a_changed:
2009-08-15 14:37:14 +00:00
self.db.commit_person(person, self.trans)
self.removed_name_format.append(person_handle)
self.progress.step()
# update the custom name name format table
for number in deleted_name_formats:
_nd.del_name_format(number)
self.db.name_formats = _nd.get_name_format(only_custom=True,
only_active=False)
def cleanup_duplicate_spouses(self):
self.progress.set_pass(_('Looking for duplicate spouses'),
self.db.get_number_of_people())
for handle in self.db.person_map.keys():
value = self.db.person_map[handle]
p = gen.lib.Person(value)
splist = p.get_family_handle_list()
if len(splist) != len(set(splist)):
new_list = []
for value in splist:
if value not in new_list:
new_list.append(value)
2009-08-15 14:37:14 +00:00
self.duplicate_links.append((handle, value))
p.set_family_handle_list(new_list)
2009-08-15 14:37:14 +00:00
self.db.commit_person(p, self.trans)
self.progress.step()
2005-12-06 06:38:09 +00:00
def fix_encoding(self):
self.progress.set_pass(_('Looking for character encoding errors'),
self.db.get_number_of_media_objects())
for handle in self.db.media_map.keys():
data = self.db.media_map[handle]
if not isinstance(data[2], unicode) or not isinstance(data[4], unicode):
2005-12-06 06:38:09 +00:00
obj = self.db.get_object_from_handle(handle)
obj.path = Utils.fix_encoding( obj.path)
obj.desc = Utils.fix_encoding( obj.desc)
2009-08-15 14:37:14 +00:00
self.db.commit_media_object(obj, self.trans)
# Once we are here, fix the mime string if not str
if not isinstance(data[3], str):
obj = self.db.get_object_from_handle(handle)
try:
if data[3] == str(data[3]):
obj.mime = str(data[3])
else:
obj.mime = ""
except:
obj.mime = ""
2009-08-15 14:37:14 +00:00
self.db.commit_media_object(obj, self.trans)
2005-12-06 06:38:09 +00:00
self.progress.step()
2002-10-20 14:25:16 +00:00
def check_for_broken_family_links(self):
# Check persons referenced by the family objects
fhandle_list = self.db.get_family_handles()
self.progress.set_pass(_('Looking for broken family links'),
len(fhandle_list) + self.db.get_number_of_people())
for family_handle in fhandle_list:
* PeopleModel.py: simplify model interface * EditPerson.py: get_family_from_handle fixes * EditSource.py: get_family_from_handle fixes * GraphLayout.py: get_family_from_handle fixes * ImageSelect.py: get_family_from_handle fixes * MediaView.py: get_family_from_handle fixes * MergeData.py: get_family_from_handle fixes * PlaceView.py: get_family_from_handle fixes * ReadXML.py: get_family_from_handle fixes * RelLib.py: get_family_from_handle fixes * Relationship.py: get_family_from_handle fixes * SelectChild.py: get_family_from_handle fixes * SourceView.py: get_family_from_handle fixes * SubstKeywords.py: get_family_from_handle fixes * WriteXML.py: get_family_from_handle fixes * gramps_main.py: get_family_from_handle fixes * plugins/AncestorChart.py: get_family_from_handle fixes * plugins/AncestorChart2.py: get_family_from_handle fixes * plugins/AncestorReport.py: get_family_from_handle fixes * plugins/Ancestors.py: get_family_from_handle fixes * plugins/Check.py: get_family_from_handle fixes * plugins/CountAncestors.py: get_family_from_handle fixes * plugins/Desbrowser.py: get_family_from_handle fixes * plugins/DescendReport.py: get_family_from_handle fixes * plugins/DetAncestralReport.py: get_family_from_handle fixes * plugins/DetDescendantReport.py: get_family_from_handle fixes * plugins/FamilyGroup.py: get_family_from_handle fixes * plugins/FanChart.py: get_family_from_handle fixes * plugins/FtmStyleAncestors.py: get_family_from_handle fixes * plugins/FtmStyleDescendants.py: get_family_from_handle fixes * plugins/GraphViz.py: get_family_from_handle fixes * plugins/IndivComplete.py: get_family_from_handle fixes * plugins/IndivSummary.py: get_family_from_handle fixes * plugins/Merge.py: get_family_from_handle fixes * plugins/RelGraph.py: get_family_from_handle fixes * plugins/Verify.py: get_family_from_handle fixes * plugins/WebPage.py: get_family_from_handle fixes * plugins/WriteCD.py: get_family_from_handle fixes * plugins/WritePkg.py: get_family_from_handle fixes * plugins/rel_de.py: get_family_from_handle fixes * plugins/rel_hu.py: get_family_from_handle fixes * plugins/rel_ru.py: get_family_from_handle fixes svn: r3443
2004-08-19 21:35:16 +00:00
family = self.db.get_family_from_handle(family_handle)
father_handle = family.get_father_handle()
mother_handle = family.get_mother_handle()
if father_handle:
father = self.db.get_person_from_handle(father_handle)
if not father:
# The person referenced by the father handle does not exist in the database
family.set_father_handle(None)
2009-08-15 14:37:14 +00:00
self.db.commit_family(family, self.trans)
self.broken_parent_links.append((father_handle, family_handle))
father_handle = None
if mother_handle:
mother = self.db.get_person_from_handle(mother_handle)
if not mother:
# The person referenced by the mother handle does not exist in the database
family.set_mother_handle(None)
2009-08-15 14:37:14 +00:00
self.db.commit_family(family, self.trans)
self.broken_parent_links.append((mother_handle, family_handle))
mother_handle = None
2002-10-20 14:25:16 +00:00
if father_handle and father and \
family_handle not in father.get_family_handle_list():
# The referenced father has no reference back to the family
2009-08-15 14:37:14 +00:00
self.broken_parent_links.append((father_handle, family_handle))
father.add_family_handle(family_handle)
2009-08-15 14:37:14 +00:00
self.db.commit_person(father, self.trans)
if mother_handle and mother and \
family_handle not in mother.get_family_handle_list():
# The referenced mother has no reference back to the family
2009-08-15 14:37:14 +00:00
self.broken_parent_links.append((mother_handle, family_handle))
mother.add_family_handle(family_handle)
2009-08-15 14:37:14 +00:00
self.db.commit_person(mother, self.trans)
2006-04-13 19:34:07 +00:00
for child_ref in family.get_child_ref_list():
child_handle = child_ref.ref
child = self.db.get_person_from_handle(child_handle)
if child:
2009-08-15 14:37:14 +00:00
if child_handle in [father_handle, mother_handle]:
# The child is one of the parents: impossible
# Remove such child from the family
family.remove_child_ref(child_ref)
2009-08-15 14:37:14 +00:00
self.db.commit_family(family, self.trans)
self.broken_links.append((child_handle, family_handle))
continue
if family_handle == child.get_main_parents_family_handle():
continue
if family_handle not in \
child.get_parent_family_handle_list():
# The referenced child has no reference to the family
2006-04-13 19:34:07 +00:00
family.remove_child_ref(child_ref)
2009-08-15 14:37:14 +00:00
self.db.commit_family(family, self.trans)
self.broken_links.append((child_handle, family_handle))
2002-10-20 14:25:16 +00:00
else:
# The person referenced by the child handle
# does not exist in the database
2006-04-13 19:34:07 +00:00
family.remove_child_ref(child_ref)
2009-08-15 14:37:14 +00:00
self.db.commit_family(family, self.trans)
self.broken_links.append((child_handle, family_handle))
new_ref_list = []
new_ref_handles = []
replace = False
for child_ref in family.get_child_ref_list():
child_handle = child_ref.ref
if child_handle in new_ref_handles:
replace = True
else:
new_ref_list.append(child_ref)
new_ref_handles.append(child_handle)
if replace:
family.set_child_ref_list(new_ref_list)
2009-08-15 14:37:14 +00:00
self.db.commit_family(family, self.trans)
self.progress.step()
# Check persons membership in referenced families
for person_handle in self.db.get_person_handles():
person = self.db.get_person_from_handle(person_handle)
phandle_list = person.get_parent_family_handle_list()
new_list = list(set(phandle_list))
if len(phandle_list) != len(new_list):
person.set_parent_family_handle_list(new_list)
2009-08-15 14:37:14 +00:00
self.db.commit_person(person, self.trans)
for par_family_handle in person.get_parent_family_handle_list():
family = self.db.get_family_from_handle(par_family_handle)
if not family:
person.remove_parent_family_handle(par_family_handle)
2009-08-15 14:37:14 +00:00
self.db.commit_person(person, self.trans)
continue
for child_handle in [child_ref.ref for child_ref
in family.get_child_ref_list()]:
if child_handle == person_handle:
break
else:
# Person is not a child in the referenced parent family
person.remove_parent_family_handle(par_family_handle)
2009-08-15 14:37:14 +00:00
self.db.commit_person(person, self.trans)
self.broken_links.append((person_handle,family_handle))
for family_handle in person.get_family_handle_list():
family = self.db.get_family_from_handle(family_handle)
if not family:
# The referenced family does not exist in database
person.remove_family_handle(family_handle)
2009-08-15 14:37:14 +00:00
self.db.commit_person(person, self.trans)
self.broken_links.append((person_handle, family_handle))
continue
if family.get_father_handle() == person_handle:
continue
if family.get_mother_handle() == person_handle:
continue
# The person is not a member of the referenced family
person.remove_family_handle(family_handle)
2009-08-15 14:37:14 +00:00
self.db.commit_person(person, self.trans)
self.broken_links.append((person_handle, family_handle))
self.progress.step()
2002-10-20 14:25:16 +00:00
2009-08-15 14:37:14 +00:00
def cleanup_missing_photos(self, cl=0):
self.progress.set_pass(_('Looking for unused objects'),
len(self.db.get_media_object_handles()))
missmedia_action = 0
#-------------------------------------------------------------------------
def remove_clicked():
# File is lost => remove all references and the object itself
for handle in self.db.get_person_handles(sort_handles=False):
person = self.db.get_person_from_handle(handle)
if person.has_media_reference(ObjectId):
person.remove_media_references([ObjectId])
2009-08-15 14:37:14 +00:00
self.db.commit_person(person, self.trans)
for handle in self.db.get_family_handles():
family = self.db.get_family_from_handle(handle)
if family.has_media_reference(ObjectId):
family.remove_media_references([ObjectId])
2009-08-15 14:37:14 +00:00
self.db.commit_family(family, self.trans)
2004-04-16 19:15:02 +00:00
for handle in self.db.get_event_handles():
event = self.db.get_event_from_handle(handle)
if event.has_media_reference(ObjectId):
event.remove_media_references([ObjectId])
self.db.commit_event(event,self.trans)
2004-04-16 19:15:02 +00:00
for handle in self.db.get_source_handles():
source = self.db.get_source_from_handle(handle)
if source.has_media_reference(ObjectId):
source.remove_media_references([ObjectId])
2009-08-15 14:37:14 +00:00
self.db.commit_source(source, self.trans)
for handle in self.db.get_place_handles():
place = self.db.get_place_from_handle(handle)
if place.has_media_reference(ObjectId):
place.remove_media_references([ObjectId])
2009-08-15 14:37:14 +00:00
self.db.commit_place(place, self.trans)
self.removed_photo.append(ObjectId)
self.db.remove_object(ObjectId,self.trans)
def leave_clicked():
2004-04-16 19:15:02 +00:00
self.bad_photo.append(ObjectId)
def select_clicked():
# File is lost => select a file to replace the lost one
def fs_close_window(obj):
2004-04-16 19:15:02 +00:00
self.bad_photo.append(ObjectId)
def fs_ok_clicked(obj):
name = Utils.get_unicode_path(fs_top.get_filename())
if os.path.isfile(name):
obj = self.db.get_object_from_handle(ObjectId)
obj.set_path(name)
2009-08-15 14:37:14 +00:00
self.db.commit_media_object(obj, self.trans)
2004-04-16 19:15:02 +00:00
self.replaced_photo.append(ObjectId)
else:
2004-04-16 19:15:02 +00:00
self.bad_photo.append(ObjectId)
fs_top = gtk.FileChooserDialog("%s - Gramps" % _("Select file"),
buttons=(gtk.STOCK_CANCEL, gtk.RESPONSE_CANCEL,
gtk.STOCK_OK, gtk.RESPONSE_OK)
)
response = fs_top.run()
if response == gtk.RESPONSE_OK:
fs_ok_clicked(fs_top)
elif response == gtk.RESPONSE_CANCEL:
fs_close_window(fs_top)
fs_top.destroy()
#-------------------------------------------------------------------------
2004-04-16 19:15:02 +00:00
for ObjectId in self.db.get_media_object_handles():
obj = self.db.get_object_from_handle(ObjectId)
photo_name = Utils.media_path_full(self.db, obj.get_path())
2005-12-06 06:38:09 +00:00
if photo_name is not None and photo_name != "" and not Utils.find_file(photo_name):
if cl:
print "Warning: media file %s was not found." \
% os.path.basename(photo_name)
2004-04-16 19:15:02 +00:00
self.bad_photo.append(ObjectId)
else:
if missmedia_action == 0:
mmd = MissingMediaDialog(_("Media object could not be found"),
_("The file:\n %(file_name)s \nis referenced in the database, but no longer exists. "
"The file may have been deleted or moved to a different location. "
"You may choose to either remove the reference from the database, "
"keep the reference to the missing file, or select a new file."
) % { 'file_name' : '<b>%s</b>' % photo_name },
remove_clicked, leave_clicked, select_clicked)
missmedia_action = mmd.default_action
elif missmedia_action == 1:
remove_clicked()
elif missmedia_action == 2:
leave_clicked()
elif missmedia_action == 3:
select_clicked()
self.progress.step()
def cleanup_empty_objects(self):
#the position of the change column in the primary objects
CHANGE_PERSON = 17
CHANGE_FAMILY = 12
CHANGE_EVENT = 10
CHANGE_SOURCE = 8
CHANGE_PLACE = 11
CHANGE_MEDIA = 8
CHANGE_REPOS = 7
CHANGE_NOTE = 5
empty_person_data = gen.lib.Person().serialize()
empty_family_data = gen.lib.Family().serialize()
empty_event_data = gen.lib.Event().serialize()
empty_source_data = gen.lib.Source().serialize()
empty_place_data = gen.lib.Place().serialize()
empty_media_data = gen.lib.MediaObject().serialize()
empty_repos_data = gen.lib.Repository().serialize()
empty_note_data = gen.lib.Note().serialize()
tables = {
'persons' : {'get_func': self.db.get_person_from_handle,
'cursor_func': self.db.get_person_cursor,
'total_func' : self.db.get_number_of_people,
'progress' : _('Looking for empty people records'),
'check_func' : lambda x : self._check_empty(x,
empty_person_data,
CHANGE_PERSON),
'remove' : self.db.remove_person},
'families': {'get_func': self.db.get_family_from_handle,
'cursor_func': self.db.get_family_cursor,
'total_func' : self.db.get_number_of_families,
'progress' : _('Looking for empty family records'),
'check_func' : lambda x : self._check_empty(x,
empty_family_data,
CHANGE_FAMILY),
'remove' : self.db.remove_family},
'events' : {'get_func': self.db.get_event_from_handle,
'cursor_func': self.db.get_event_cursor,
'total_func' : self.db.get_number_of_events,
'progress' : _('Looking for empty event records'),
'check_func' : lambda x : self._check_empty(x,
empty_event_data,
CHANGE_EVENT),
'remove' : self.db.remove_event},
'sources' : {'get_func': self.db.get_source_from_handle,
'cursor_func': self.db.get_source_cursor,
'total_func' : self.db.get_number_of_sources,
'progress' : _('Looking for empty source records'),
'check_func' : lambda x : self._check_empty(x,
empty_source_data,
CHANGE_SOURCE),
'remove' : self.db.remove_source},
'places' : {'get_func': self.db.get_place_from_handle,
'cursor_func': self.db.get_place_cursor,
'total_func' : self.db.get_number_of_places,
'progress' : _('Looking for empty place records'),
'check_func' : lambda x : self._check_empty(x,
empty_place_data,
CHANGE_PLACE),
'remove' : self.db.remove_place},
'media' : {'get_func': self.db.get_object_from_handle,
'cursor_func': self.db.get_media_cursor,
'progress' : _('Looking for empty media records'),
'total_func' : self.db.get_number_of_media_objects,
'check_func' : lambda x : self._check_empty(x,
empty_media_data,
CHANGE_MEDIA),
'remove' : self.db.remove_object},
'repos' : {'get_func': self.db.get_repository_from_handle,
'cursor_func': self.db.get_repository_cursor,
'total_func' : self.db.get_number_of_repositories,
'progress' : _('Looking for empty repository records'),
'check_func' : lambda x : self._check_empty(x,
empty_repos_data,
CHANGE_REPOS),
'remove' : self.db.remove_repository},
'notes' : {'get_func': self.db.get_note_from_handle,
'cursor_func': self.db.get_note_cursor,
'total_func' : self.db.get_number_of_notes,
'progress' : _('Looking for empty note records'),
'check_func' : lambda x : self._check_empty(x,
empty_note_data,
CHANGE_NOTE),
'remove' : self.db.remove_note},
}
for the_type, the_func in tables.iteritems():
with the_func['cursor_func']() as cursor:
total = the_func['total_func']()
check = the_func['check_func']
remove_func = the_func['remove']
self.progress.set_pass(the_func['progress'],total)
for handle, data in cursor:
self.progress.step()
if check(data):
self.empty_objects[the_type].append(handle)
#we cannot remove here as that would destroy cursor
#now remove
for handle in self.empty_objects[the_type]:
remove_func(handle, self.trans)
def _check_empty(self, data, empty_data, changepos):
"""compare the data with the data of an empty object
change, handle and gramps_id are not compared """
if changepos is not None:
return (data[2:changepos] == empty_data[2:changepos] and
data[changepos+1:] == empty_data[changepos+1:]
)
else :
return data[2:] == empty_data[2:]
2002-10-20 14:25:16 +00:00
2009-08-15 14:37:14 +00:00
def cleanup_empty_families(self, automatic):
fhandle_list = self.db.get_family_handles()
self.progress.set_pass(_('Looking for empty families'),
len(fhandle_list))
for family_handle in fhandle_list:
self.progress.step()
* PeopleModel.py: simplify model interface * EditPerson.py: get_family_from_handle fixes * EditSource.py: get_family_from_handle fixes * GraphLayout.py: get_family_from_handle fixes * ImageSelect.py: get_family_from_handle fixes * MediaView.py: get_family_from_handle fixes * MergeData.py: get_family_from_handle fixes * PlaceView.py: get_family_from_handle fixes * ReadXML.py: get_family_from_handle fixes * RelLib.py: get_family_from_handle fixes * Relationship.py: get_family_from_handle fixes * SelectChild.py: get_family_from_handle fixes * SourceView.py: get_family_from_handle fixes * SubstKeywords.py: get_family_from_handle fixes * WriteXML.py: get_family_from_handle fixes * gramps_main.py: get_family_from_handle fixes * plugins/AncestorChart.py: get_family_from_handle fixes * plugins/AncestorChart2.py: get_family_from_handle fixes * plugins/AncestorReport.py: get_family_from_handle fixes * plugins/Ancestors.py: get_family_from_handle fixes * plugins/Check.py: get_family_from_handle fixes * plugins/CountAncestors.py: get_family_from_handle fixes * plugins/Desbrowser.py: get_family_from_handle fixes * plugins/DescendReport.py: get_family_from_handle fixes * plugins/DetAncestralReport.py: get_family_from_handle fixes * plugins/DetDescendantReport.py: get_family_from_handle fixes * plugins/FamilyGroup.py: get_family_from_handle fixes * plugins/FanChart.py: get_family_from_handle fixes * plugins/FtmStyleAncestors.py: get_family_from_handle fixes * plugins/FtmStyleDescendants.py: get_family_from_handle fixes * plugins/GraphViz.py: get_family_from_handle fixes * plugins/IndivComplete.py: get_family_from_handle fixes * plugins/IndivSummary.py: get_family_from_handle fixes * plugins/Merge.py: get_family_from_handle fixes * plugins/RelGraph.py: get_family_from_handle fixes * plugins/Verify.py: get_family_from_handle fixes * plugins/WebPage.py: get_family_from_handle fixes * plugins/WriteCD.py: get_family_from_handle fixes * plugins/WritePkg.py: get_family_from_handle fixes * plugins/rel_de.py: get_family_from_handle fixes * plugins/rel_hu.py: get_family_from_handle fixes * plugins/rel_ru.py: get_family_from_handle fixes svn: r3443
2004-08-19 21:35:16 +00:00
family = self.db.get_family_from_handle(family_handle)
family_id = family.get_gramps_id()
father_handle = family.get_father_handle()
mother_handle = family.get_mother_handle()
if not father_handle and not mother_handle and \
len(family.get_child_ref_list()) == 0:
self.empty_family.append(family_id)
self.delete_empty_family(family_handle)
2002-10-20 14:25:16 +00:00
2009-08-15 14:37:14 +00:00
def delete_empty_family(self, family_handle):
for key in self.db.get_person_handles(sort_handles=False):
child = self.db.get_person_from_handle(key)
child.remove_parent_family_handle(family_handle)
child.remove_family_handle(family_handle)
2009-08-15 14:37:14 +00:00
self.db.remove_family(family_handle, self.trans)
2002-10-20 14:25:16 +00:00
def check_parent_relationships(self):
"""Repair father=female or mother=male in hetero families
"""
fhandle_list = self.db.get_family_handles()
self.progress.set_pass(_('Looking for broken parent relationships'),
len(fhandle_list))
for family_handle in fhandle_list:
2005-12-06 06:38:09 +00:00
self.progress.step()
family = self.db.get_family_from_handle(family_handle)
father_handle = family.get_father_handle()
if father_handle:
fgender = self.db.get_person_from_handle(father_handle
).get_gender()
else:
fgender = None
mother_handle = family.get_mother_handle()
if mother_handle:
mgender = self.db.get_person_from_handle(mother_handle
).get_gender()
2002-10-20 14:25:16 +00:00
else:
mgender = None
if (fgender == gen.lib.Person.FEMALE \
or mgender == gen.lib.Person.MALE) \
and fgender != mgender:
# swap. note: (at most) one handle may be None
family.set_father_handle(mother_handle)
family.set_mother_handle(father_handle)
2009-08-15 14:37:14 +00:00
self.db.commit_family(family, self.trans)
self.fam_rel.append(family_handle)
2002-10-20 14:25:16 +00:00
def check_events(self):
self.progress.set_pass(_('Looking for event problems'),
self.db.get_number_of_people()
+self.db.get_number_of_families())
for key in self.db.get_person_handles(sort_handles=False):
self.progress.step()
person = self.db.get_person_from_handle(key)
2006-03-30 17:29:30 +00:00
birth_ref = person.get_birth_ref()
if birth_ref:
birth_handle = birth_ref.ref
birth = self.db.get_event_from_handle(birth_handle)
if not birth:
# The birth event referenced by the birth handle
# does not exist in the database
2006-03-30 17:29:30 +00:00
person.set_birth_ref(None)
2009-08-15 14:37:14 +00:00
self.db.commit_person(person, self.trans)
self.invalid_events.append(key)
else:
if int(birth.get_type()) != gen.lib.EventType.BIRTH:
# Birth event was not of the type "Birth"
birth.set_type(gen.lib.EventType(gen.lib.EventType.BIRTH))
2009-08-15 14:37:14 +00:00
self.db.commit_event(birth, self.trans)
self.invalid_birth_events.append(key)
2006-03-30 17:29:30 +00:00
death_ref = person.get_death_ref()
if death_ref:
death_handle = death_ref.ref
death = self.db.get_event_from_handle(death_handle)
if not death:
# The death event referenced by the death handle
# does not exist in the database
2006-03-30 17:29:30 +00:00
person.set_death_ref(None)
2009-08-15 14:37:14 +00:00
self.db.commit_person(person, self.trans)
self.invalid_events.append(key)
else:
if int(death.get_type()) != gen.lib.EventType.DEATH:
# Death event was not of the type "Death"
death.set_type(gen.lib.EventType(gen.lib.EventType.DEATH))
2009-08-15 14:37:14 +00:00
self.db.commit_event(death, self.trans)
self.invalid_death_events.append(key)
2006-03-30 17:29:30 +00:00
if person.get_event_ref_list():
for event_ref in person.get_event_ref_list():
event_handle = event_ref.ref
event = self.db.get_event_from_handle(event_handle)
if not event:
# The event referenced by the person
# does not exist in the database
#TODO: There is no better way?
2006-03-30 17:29:30 +00:00
person.get_event_ref_list().remove(event_ref)
self.db.commit_person(person,self.trans)
self.invalid_events.append(key)
elif not isinstance(person.get_event_ref_list(), list):
# event_list is None or other garbage
2006-03-30 17:29:30 +00:00
person.set_event_ref_list([])
2009-08-15 14:37:14 +00:00
self.db.commit_person(person, self.trans)
self.invalid_events.append(key)
for key in self.db.get_family_handles():
self.progress.step()
family = self.db.get_family_from_handle(key)
2006-03-30 17:29:30 +00:00
if family.get_event_ref_list():
for event_ref in family.get_event_ref_list():
event_handle = event_ref.ref
event = self.db.get_event_from_handle(event_handle)
if not event:
# The event referenced by the family
# does not exist in the database
print family.gramps_id
nlist = [ x for x in family.get_event_ref_list() \
if x.ref != event_handle]
family.set_event_ref_list(nlist)
2009-08-15 14:37:14 +00:00
self.db.commit_family(family, self.trans)
self.invalid_events.append(key)
elif not isinstance(family.get_event_ref_list(), list):
# event_list is None or other garbage
2006-03-30 17:29:30 +00:00
family.set_event_ref_list([])
2009-08-15 14:37:14 +00:00
self.db.commit_family(family, self.trans)
self.invalid_events.append(key)
def check_person_references(self):
plist = self.db.get_person_handles()
self.progress.set_pass(_('Looking for person reference problems'),
len(plist))
for key in plist:
person = self.db.get_person_from_handle(key)
for pref in person.get_person_ref_list():
p = self.db.get_person_from_handle( pref.ref)
if not p:
# The referenced person does not exist in the database
person.get_person_ref_list().remove(pref)
2009-08-15 14:37:14 +00:00
self.db.commit_person(person, self.trans)
self.invalid_person_references.append(key)
def check_dates(self):
"""
Fix issues in 3.1.0 upgrade: missed some dates on associated people
source dates.
"""
plist = self.db.get_person_handles()
self.progress.set_pass(_('Checking people for proper date formats'),
len(plist))
# First, decode all of a person:
for handle in plist:
need_to_fix = False
person = self.db.person_map[handle]
(phandle, # 0
gramps_id, # 1
gender, # 2
primary_name, # 3
alternate_names, # 4
death_ref_index, # 5
birth_ref_index, # 6
event_ref_list, # 7
family_list, # 8
parent_family_list, # 9
media_list, # 10
address_list, # 11
attribute_list, # 12
urls, # 13
lds_ord_list, # 14
psource_list, # 15
pnote_list, # 16
change, # 17
marker, # 18
pprivate, # 19
person_ref_list, # 20
) = person
# Take apart person reference list:
new_person_ref_list = []
for person_ref in person_ref_list:
(private, source_list, note_list, pref, rel) = person_ref
new_source_list = []
# Check dates to see if correct format:
for source in source_list:
(date, private, note_list, confidence, sref, page) = source
if date is None:
new_date = None
else:
if len(date) == 7:
# This is correct:
(calendar, modifier, quality, dateval, text, sortval,
newyear) = date
elif len(date) == 6:
# This is necessary to fix 3.1.0 bug:
(calendar, modifier, quality, dateval, text, sortval) = date
newyear = 0
need_to_fix = True
else:
# FIXME: What to do with an invalid date?
# Make a new one?
(calendar, modifier, quality, dateval, text, sortval,
newyear) = gen.lib.Date().serialize()
need_to_fix = True
# Put date back together:
new_date = (calendar, modifier, quality, dateval, text, sortval,
newyear)
# Put source_list together:
new_source_list.append((new_date, private, note_list, confidence, sref, page))
# Put new_person_ref together:
new_person_ref_list.append((private, new_source_list, note_list, pref, rel))
# Put person back together:
if need_to_fix:
new_person = (phandle, # 0
gramps_id, # 1
gender, # 2
primary_name, # 3
alternate_names, # 4
death_ref_index, # 5
birth_ref_index, # 6
event_ref_list, # 7
family_list, # 8
parent_family_list, # 9
media_list, # 10
address_list, # 11
attribute_list, # 12
urls, # 13
lds_ord_list, # 14
psource_list, # 15
pnote_list, # 16
change, # 17
marker, # 18
pprivate, # 19
new_person_ref_list, # 20
)
p = gen.lib.Person(new_person)
2009-08-15 14:37:14 +00:00
self.db.commit_person(p, self.trans)
self.invalid_dates.append(handle)
self.progress.step()
flist = self.db.get_family_handles()
self.progress.set_pass(_('Checking families for proper date formats'),
len(flist))
# First, decode all of a person:
for handle in flist:
need_to_fix = False
family = self.db.family_map[handle]
(fhandle, gramps_id, father_handle, mother_handle,
child_ref_list, the_type, event_ref_list, media_list,
attribute_list, lds_seal_list, source_list, note_list,
change, marker, private) = family
new_child_ref_list = []
for data in child_ref_list:
(private, source_list, note_list, ref, frel, mrel) = data
new_source_list = []
# Check dates to see if correct format:
for source in source_list:
(date, private, note_list, confidence, sref, page) = source
if date is None:
new_date = None
else:
if len(date) == 7:
# This is correct:
(calendar, modifier, quality, dateval, text, sortval,
newyear) = date
elif len(date) == 6:
# This is necessary to fix 3.1.0 bug:
(calendar, modifier, quality, dateval, text, sortval) = date
newyear = 0
need_to_fix = True
else:
# FIXME: What to do with an invalid date?
# Make a new one?
(calendar, modifier, quality, dateval, text, sortval,
newyear) = gen.lib.Date().serialize()
need_to_fix = True
# Put date back together:
new_date = (calendar, modifier, quality, dateval, text, sortval,
newyear)
# Put source_list together:
new_source_list.append((new_date, private, note_list, confidence, sref, page))
# Put child ref list together:
new_child_ref_list.append((private, new_source_list, note_list, ref, frel, mrel))
if need_to_fix:
new_family = (fhandle, gramps_id, father_handle, mother_handle,
new_child_ref_list, the_type, event_ref_list, media_list,
attribute_list, lds_seal_list, source_list, note_list,
change, marker, private)
f = gen.lib.Family()
f.unserialize(new_family)
2009-08-15 14:37:14 +00:00
self.db.commit_family(f, self.trans)
self.invalid_dates.append(handle)
self.progress.step()
def check_repo_references(self):
slist = self.db.get_source_handles()
self.progress.set_pass(_('Looking for repository reference problems'),
len(slist))
for key in slist:
source = self.db.get_source_from_handle(key)
for reporef in source.get_reporef_list():
r = self.db.get_repository_from_handle(reporef.ref)
if not r:
# The referenced repository does not exist in the database
source.get_reporef_list().remove(reporef)
2009-08-15 14:37:14 +00:00
self.db.commit_source(source, self.trans)
self.invalid_repo_references.append(key)
def check_place_references(self):
plist = self.db.get_person_handles()
flist = self.db.get_family_handles()
elist = self.db.get_event_handles()
self.progress.set_pass(_('Looking for place reference problems'),
len(elist)+len(plist)+len(flist))
# check persons -> the LdsOrd references a place
for key in plist:
person = self.db.get_person_from_handle(key)
for ordinance in person.lds_ord_list:
place_handle = ordinance.get_place_handle()
if place_handle:
place = self.db.get_place_from_handle(place_handle)
if not place:
# The referenced place does not exist in the database
ordinance.set_place_handle("")
2009-08-15 14:37:14 +00:00
self.db.commit_person(person, self.trans)
self.invalid_place_references.append(key)
# check families -> the LdsOrd references a place
for key in flist:
family = self.db.get_family_from_handle(key)
for ordinance in family.lds_ord_list:
place_handle = ordinance.get_place_handle()
if place_handle:
place = self.db.get_place_from_handle(place_handle)
if not place:
# The referenced place does not exist in the database
ordinance.set_place_handle("")
2009-08-15 14:37:14 +00:00
self.db.commit_family(family, self.trans)
self.invalid_place_references.append(key)
# check events
for key in elist:
event = self.db.get_event_from_handle(key)
place_handle = event.get_place_handle()
if place_handle:
place = self.db.get_place_from_handle(place_handle)
if not place:
# The referenced place does not exist in the database
event.set_place_handle("")
2009-08-15 14:37:14 +00:00
self.db.commit_event(event, self.trans)
self.invalid_place_references.append(key)
def check_source_references(self):
known_handles = self.db.get_source_handles()
total = (
self.db.get_number_of_people() +
self.db.get_number_of_families() +
self.db.get_number_of_events() +
self.db.get_number_of_places() +
self.db.get_number_of_media_objects() +
self.db.get_number_of_sources() +
self.db.get_number_of_repositories()
)
self.progress.set_pass(_('Looking for source reference problems'),
total)
for handle in self.db.person_map.keys():
self.progress.step()
info = self.db.person_map[handle]
person = gen.lib.Person()
person.unserialize(info)
handle_list = person.get_referenced_handles_recursively()
bad_handles = [ item[1] for item in handle_list
if item[0] == 'Source' and
item[1] not in known_handles ]
if bad_handles:
person.remove_source_references(bad_handles)
self.db.commit_person(person,self.trans)
new_bad_handles = [handle for handle in bad_handles if handle
not in self.invalid_source_references]
self.invalid_source_references += new_bad_handles
for handle in self.db.family_map.keys():
self.progress.step()
info = self.db.family_map[handle]
family = gen.lib.Family()
family.unserialize(info)
handle_list = family.get_referenced_handles_recursively()
bad_handles = [ item[1] for item in handle_list
if item[0] == 'Source' and
item[1] not in known_handles ]
if bad_handles:
family.remove_source_references(bad_handles)
2009-08-15 14:37:14 +00:00
self.db.commit_family(family, self.trans)
new_bad_handles = [handle for handle in bad_handles if handle
not in self.invalid_source_references]
self.invalid_source_references += new_bad_handles
for handle in self.db.place_map.keys():
self.progress.step()
info = self.db.place_map[handle]
place = gen.lib.Place()
place.unserialize(info)
handle_list = place.get_referenced_handles_recursively()
bad_handles = [ item[1] for item in handle_list
if item[0] == 'Source' and
item[1] not in known_handles ]
if bad_handles:
place.remove_source_references(bad_handles)
self.db.commit_place(place,self.trans)
new_bad_handles = [handle for handle in bad_handles if handle
not in self.invalid_source_references]
self.invalid_source_references += new_bad_handles
for handle in self.db.repository_map.keys():
self.progress.step()
info = self.db.repository_map[handle]
repo = gen.lib.Repository()
repo.unserialize(info)
handle_list = repo.get_referenced_handles_recursively()
bad_handles = [ item[1] for item in handle_list
if item[0] == 'Source' and
item[1] not in known_handles ]
if bad_handles:
repo.remove_source_references(bad_handles)
2009-08-15 14:37:14 +00:00
self.db.commit_repository(repo, self.trans)
new_bad_handles = [handle for handle in bad_handles if handle
not in self.invalid_source_references]
self.invalid_source_references += new_bad_handles
#I think this for loop is useless! sources in sources map exist...
for handle in known_handles:
self.progress.step()
info = self.db.source_map[handle]
source = gen.lib.Source()
source.unserialize(info)
handle_list = source.get_referenced_handles_recursively()
bad_handles = [ item[1] for item in handle_list
if item[0] == 'Source' and
item[1] not in known_handles ]
if bad_handles:
source.remove_source_references(bad_handles)
2009-08-15 14:37:14 +00:00
self.db.commit_source(source, self.trans)
new_bad_handles = [handle for handle in bad_handles if handle
not in self.invalid_source_references]
self.invalid_source_references += new_bad_handles
for handle in self.db.media_map.keys():
self.progress.step()
info = self.db.media_map[handle]
obj = gen.lib.MediaObject()
obj.unserialize(info)
handle_list = obj.get_referenced_handles_recursively()
bad_handles = [ item[1] for item in handle_list
if item[0] == 'Source' and
item[1] not in known_handles ]
if bad_handles:
obj.remove_source_references(bad_handles)
2009-08-15 14:37:14 +00:00
self.db.commit_media_object(obj, self.trans)
new_bad_handles = [handle for handle in bad_handles if handle
not in self.invalid_source_references]
self.invalid_source_references += new_bad_handles
for handle in self.db.event_map.keys():
self.progress.step()
info = self.db.event_map[handle]
event = gen.lib.Event()
event.unserialize(info)
handle_list = event.get_referenced_handles_recursively()
bad_handles = [ item[1] for item in handle_list
if item[0] == 'Source' and
item[1] not in known_handles ]
if bad_handles:
event.remove_source_references(bad_handles)
2009-08-15 14:37:14 +00:00
self.db.commit_event(event, self.trans)
new_bad_handles = [handle for handle in bad_handles if handle
not in self.invalid_source_references]
self.invalid_source_references += new_bad_handles
def check_media_references(self):
known_handles = self.db.get_media_object_handles(False)
total = (
self.db.get_number_of_people() +
self.db.get_number_of_families() +
self.db.get_number_of_events() +
self.db.get_number_of_places() +
self.db.get_number_of_sources()
)
self.progress.set_pass(_('Looking for media object reference problems'),
total)
for handle in self.db.person_map.keys():
self.progress.step()
info = self.db.person_map[handle]
person = gen.lib.Person()
person.unserialize(info)
handle_list = person.get_referenced_handles_recursively()
bad_handles = [ item[1] for item in handle_list
if item[0] == 'MediaObject' and
item[1] not in known_handles ]
if bad_handles:
person.remove_media_references(bad_handles)
2009-08-15 14:37:14 +00:00
self.db.commit_person(person, self.trans)
new_bad_handles = [handle for handle in bad_handles if handle
not in self.invalid_media_references]
self.invalid_media_references += new_bad_handles
for handle in self.db.family_map.keys():
self.progress.step()
info = self.db.family_map[handle]
family = gen.lib.Family()
family.unserialize(info)
handle_list = family.get_referenced_handles_recursively()
bad_handles = [ item[1] for item in handle_list
if item[0] == 'MediaObject' and
item[1] not in known_handles ]
if bad_handles:
family.remove_media_references(bad_handles)
2009-08-15 14:37:14 +00:00
self.db.commit_family(family, self.trans)
new_bad_handles = [handle for handle in bad_handles if handle
not in self.invalid_media_references]
self.invalid_media_references += new_bad_handles
for handle in self.db.place_map.keys():
self.progress.step()
info = self.db.place_map[handle]
place = gen.lib.Place()
place.unserialize(info)
handle_list = place.get_referenced_handles_recursively()
bad_handles = [ item[1] for item in handle_list
if item[0] == 'MediaObject' and
item[1] not in known_handles ]
if bad_handles:
place.remove_media_references(bad_handles)
2009-08-15 14:37:14 +00:00
self.db.commit_place(place, self.trans)
new_bad_handles = [handle for handle in bad_handles if handle
not in self.invalid_media_references]
self.invalid_media_references += new_bad_handles
for handle in self.db.event_map.keys():
self.progress.step()
info = self.db.event_map[handle]
event = gen.lib.Event()
event.unserialize(info)
handle_list = event.get_referenced_handles_recursively()
bad_handles = [ item[1] for item in handle_list
if item[0] == 'MediaObject' and
item[1] not in known_handles ]
if bad_handles:
event.remove_media_references(bad_handles)
2009-08-15 14:37:14 +00:00
self.db.commit_event(event, self.trans)
new_bad_handles = [handle for handle in bad_handles if handle
not in self.invalid_media_references]
self.invalid_media_references += new_bad_handles
for handle in self.db.source_map.keys():
self.progress.step()
info = self.db.source_map[handle]
source = gen.lib.Source()
source.unserialize(info)
handle_list = source.get_referenced_handles_recursively()
bad_handles = [ item[1] for item in handle_list
if item[0] == 'MediaObject' and
item[1] not in known_handles ]
if bad_handles:
source.remove_media_references(bad_handles)
2009-08-15 14:37:14 +00:00
self.db.commit_source(source, self.trans)
new_bad_handles = [handle for handle in bad_handles if handle
not in self.invalid_media_references]
self.invalid_media_references += new_bad_handles
def check_note_references(self):
known_handles = self.db.get_note_handles()
total = (
self.db.get_number_of_people() +
self.db.get_number_of_families() +
self.db.get_number_of_events() +
self.db.get_number_of_places() +
self.db.get_number_of_media_objects() +
self.db.get_number_of_sources() +
self.db.get_number_of_repositories()
)
self.progress.set_pass(_('Looking for note reference problems'),
total)
for handle in self.db.person_map.keys():
self.progress.step()
info = self.db.person_map[handle]
person = gen.lib.Person()
person.unserialize(info)
handle_list = person.get_referenced_handles_recursively()
bad_handles = [ item[1] for item in handle_list
if item[0] == 'Note' and
item[1] not in known_handles ]
for bad_handle in bad_handles:
person.remove_note(bad_handle)
if bad_handles:
2009-08-15 14:37:14 +00:00
self.db.commit_person(person, self.trans)
new_bad_handles = [handle for handle in bad_handles if handle
not in self.invalid_note_references]
self.invalid_note_references += new_bad_handles
for handle in self.db.family_map.keys():
self.progress.step()
info = self.db.family_map[handle]
family = gen.lib.Family()
family.unserialize(info)
handle_list = family.get_referenced_handles_recursively()
bad_handles = [ item[1] for item in handle_list
if item[0] == 'Note' and
item[1] not in known_handles ]
for bad_handle in bad_handles:
family.remove_note(bad_handle)
if bad_handles:
2009-08-15 14:37:14 +00:00
self.db.commit_family(family, self.trans)
new_bad_handles = [handle for handle in bad_handles if handle
not in self.invalid_note_references]
self.invalid_note_references += new_bad_handles
for handle in self.db.place_map.keys():
self.progress.step()
info = self.db.place_map[handle]
place = gen.lib.Place()
place.unserialize(info)
handle_list = place.get_referenced_handles_recursively()
bad_handles = [ item[1] for item in handle_list
if item[0] == 'Note' and
item[1] not in known_handles ]
for bad_handle in bad_handles:
place.remove_note(bad_handle)
if bad_handles:
2009-08-15 14:37:14 +00:00
self.db.commit_place(place, self.trans)
new_bad_handles = [handle for handle in bad_handles if handle
not in self.invalid_note_references]
self.invalid_note_references += new_bad_handles
for handle in self.db.source_map.keys():
self.progress.step()
info = self.db.source_map[handle]
source = gen.lib.Source()
source.unserialize(info)
handle_list = source.get_referenced_handles_recursively()
bad_handles = [ item[1] for item in handle_list
if item[0] == 'Note' and
item[1] not in known_handles ]
for bad_handle in bad_handles:
source.remove_note(bad_handle)
if bad_handles:
2009-08-15 14:37:14 +00:00
self.db.commit_source(source, self.trans)
new_bad_handles = [handle for handle in bad_handles if handle
not in self.invalid_note_references]
self.invalid_note_references += new_bad_handles
for handle in self.db.media_map.keys():
self.progress.step()
info = self.db.media_map[handle]
obj = gen.lib.MediaObject()
obj.unserialize(info)
handle_list = obj.get_referenced_handles_recursively()
bad_handles = [ item[1] for item in handle_list
if item[0] == 'Note' and
item[1] not in known_handles ]
for bad_handle in bad_handles:
obj.remove_note(bad_handle)
if bad_handles:
2009-08-15 14:37:14 +00:00
self.db.commit_media_object(obj, self.trans)
new_bad_handles = [handle for handle in bad_handles if handle
not in self.invalid_note_references]
self.invalid_note_references += new_bad_handles
for handle in self.db.event_map.keys():
self.progress.step()
info = self.db.event_map[handle]
event = gen.lib.Event()
event.unserialize(info)
handle_list = event.get_referenced_handles_recursively()
bad_handles = [ item[1] for item in handle_list
if item[0] == 'Note' and
item[1] not in known_handles ]
for bad_handle in bad_handles:
event.remove_note(bad_handle)
if bad_handles:
2009-08-15 14:37:14 +00:00
self.db.commit_event(event, self.trans)
new_bad_handles = [handle for handle in bad_handles if handle
not in self.invalid_note_references]
self.invalid_note_references += new_bad_handles
for handle in self.db.repository_map.keys():
self.progress.step()
info = self.db.repository_map[handle]
repo = gen.lib.Repository()
repo.unserialize(info)
handle_list = repo.get_referenced_handles_recursively()
bad_handles = [ item[1] for item in handle_list
if item[0] == 'Note' and
item[1] not in known_handles ]
for bad_handle in bad_handles:
repo.remove_note(bad_handle)
if bad_handles:
2009-08-15 14:37:14 +00:00
self.db.commit_repository(repo, self.trans)
new_bad_handles = [handle for handle in bad_handles if handle
not in self.invalid_note_references]
self.invalid_note_references += new_bad_handles
2009-08-15 14:37:14 +00:00
def build_report(self, cl=0):
self.progress.close()
bad_photos = len(self.bad_photo)
replaced_photos = len(self.replaced_photo)
removed_photos = len(self.removed_photo)
photos = bad_photos + replaced_photos + removed_photos
2002-10-20 14:25:16 +00:00
efam = len(self.empty_family)
blink = len(self.broken_links)
plink = len(self.broken_parent_links)
slink = len(self.duplicate_links)
2002-10-20 14:25:16 +00:00
rel = len(self.fam_rel)
event_invalid = len(self.invalid_events)
birth_invalid = len(self.invalid_birth_events)
death_invalid = len(self.invalid_death_events)
person = birth_invalid + death_invalid
person_references = len(self.invalid_person_references)
invalid_dates = len(self.invalid_dates)
place_references = len(self.invalid_place_references)
source_references = len(self.invalid_source_references)
repo_references = len(self.invalid_repo_references)
media_references = len(self.invalid_media_references)
note_references = len(self.invalid_note_references)
name_format = len(self.removed_name_format)
empty_objs = (
len(self.empty_objects['persons']) +
len(self.empty_objects['families']) +
len(self.empty_objects['events']) +
len(self.empty_objects['sources']) +
len(self.empty_objects['media']) +
len(self.empty_objects['places']) +
len(self.empty_objects['repos']) +
len(self.empty_objects['notes'])
)
errors = (photos + efam + blink + plink + slink + rel +
event_invalid + person +
person_references + place_references + source_references +
repo_references + media_references + note_references +
name_format + empty_objs + invalid_dates
)
2002-10-20 14:25:16 +00:00
if errors == 0:
2003-05-19 04:54:14 +00:00
if cl:
print "No errors were found: the database has passed internal checks."
else:
OkDialog(_("No errors were found"),
_('The database has passed internal checks'))
2003-05-19 04:54:14 +00:00
return 0
2002-10-20 14:25:16 +00:00
2003-05-19 04:54:14 +00:00
self.text = cStringIO.StringIO()
2002-10-20 14:25:16 +00:00
if blink > 0:
2009-02-03 10:33:19 +00:00
self.text.write(ngettext("%d broken child/family link was fixed\n", \
2009-01-31 11:29:44 +00:00
"%d broken child-family links were found\n", blink) % blink)
2009-08-15 14:37:14 +00:00
for (person_handle, family_handle) in self.broken_links:
person = self.db.get_person_from_handle(person_handle)
if person:
cn = person.get_primary_name().get_name()
else:
cn = _("Non existing child")
try:
family = self.db.get_family_from_handle(family_handle)
2009-08-15 14:37:14 +00:00
pn = Utils.family_name(family, self.db)
except:
pn = _("Unknown")
2003-05-19 04:54:14 +00:00
self.text.write('\t')
2009-08-15 14:37:14 +00:00
self.text.write(_("%s was removed from the family of %s\n") % (cn, pn))
2002-10-20 14:25:16 +00:00
if plink > 0:
2009-02-03 10:33:19 +00:00
self.text.write(ngettext("%d broken spouse/family link was fixed\n", \
2009-01-31 11:29:44 +00:00
"%d broken spouse/family links were found\n", plink) % plink)
2009-08-15 14:37:14 +00:00
for (person_handle, family_handle) in self.broken_parent_links:
person = self.db.get_person_from_handle(person_handle)
if person:
cn = person.get_primary_name().get_name()
else:
cn = _("Non existing person")
family = self.db.get_family_from_handle(family_handle)
if family:
2009-08-15 14:37:14 +00:00
pn = Utils.family_name(family, self.db)
else:
pn = family_handle
self.text.write('\t')
2009-08-15 14:37:14 +00:00
self.text.write(_("%s was restored to the family of %s\n") % (cn, pn))
2002-10-20 14:25:16 +00:00
if slink > 0:
2009-02-03 10:33:19 +00:00
self.text.write(ngettext("%d duplicate spouse/family link was found\n", \
2009-01-31 11:29:44 +00:00
"%d duplicate spouse/family links were found\n", slink) % slink)
2009-08-15 14:37:14 +00:00
for (person_handle, family_handle) in self.broken_parent_links:
person = self.db.get_person_from_handle(person_handle)
if person:
cn = person.get_primary_name().get_name()
else:
cn = _("Non existing person")
family = self.db.get_family_from_handle(family_handle)
if family:
2009-08-15 14:37:14 +00:00
pn = Utils.family_name(family, self.db)
else:
2005-12-06 06:38:09 +00:00
pn = _("None")
self.text.write('\t')
2009-08-15 14:37:14 +00:00
self.text.write(_("%s was restored to the family of %s\n") % (cn, pn))
2002-10-20 14:25:16 +00:00
if efam == 1:
2009-02-03 10:33:19 +00:00
self.text.write(_("%d family with no parents or children found, removed.\n"))
2005-12-06 06:38:09 +00:00
self.text.write("\t%s\n" % self.empty_family[0])
2002-10-20 14:25:16 +00:00
elif efam > 1:
self.text.write(_("%d families with no parents or children, removed.\n") % efam)
2009-01-31 11:29:44 +00:00
if rel:
2009-02-03 10:33:19 +00:00
self.text.write(ngettext("%d corrupted family relationship fixed\n", \
2009-01-31 11:29:44 +00:00
"%d corrupted family relationship fixed\n", rel) % rel )
2009-01-31 11:29:44 +00:00
if person_references:
2009-02-03 10:33:19 +00:00
self.text.write(ngettext("%d person was referenced but not found\n", \
2009-01-31 11:29:44 +00:00
"%d persons were referenced, but not found\n", person_references) % person_references)
if invalid_dates:
self.text.write(ngettext("%d date was corrected\n", \
"%d dates were corrected\n", invalid_dates) % invalid_dates)
2009-01-31 11:29:44 +00:00
if repo_references:
2009-02-03 10:33:19 +00:00
self.text.write(ngettext("%d repository was referenced but not found\n", \
2009-01-31 11:29:44 +00:00
"%d repositories were referenced, but not found\n", repo_references) % repo_references)
if photos:
2009-02-03 10:33:19 +00:00
self.text.write(ngettext("%d media object was referenced, but not found\n", \
2009-01-31 11:29:44 +00:00
"%d media objects were referenced, but not found\n", photos) % photos)
if bad_photos:
2009-02-03 10:33:19 +00:00
self.text.write(ngettext("Reference to %d missing media object was kept\n", \
2009-01-31 11:29:44 +00:00
"References to %d media objects were kept\n", bad_photos) % bad_photos)
if replaced_photos:
2009-02-03 10:33:19 +00:00
self.text.write(ngettext("%d missing media object was replaced\n", \
2009-01-31 11:29:44 +00:00
"%d missing media objects were replaced\n", replaced_photos) % replaced_photos)
if removed_photos:
2009-02-03 10:33:19 +00:00
self.text.write(ngettext("%d missing media object was removed\n", \
2009-01-31 11:29:44 +00:00
"%d missing media objects were removed\n", removed_photos) % removed_photos)
if event_invalid:
2009-02-03 10:33:19 +00:00
self.text.write(ngettext("%d invalid event reference was removed\n", \
2009-01-31 11:29:44 +00:00
"%d invalid event references were removed\n", event_invalid) % event_invalid)
if birth_invalid:
2009-02-03 10:33:19 +00:00
self.text.write(ngettext("%d invalid birth event name was fixed\n", \
2009-01-31 11:29:44 +00:00
"%d invalid birth event names were fixed\n", birth_invalid) % birth_invalid)
if death_invalid:
2009-02-03 10:33:19 +00:00
self.text.write(ngettext("%d invalid death event name was fixed\n", \
2009-01-31 11:29:44 +00:00
"%d invalid death event names were fixed\n", death_invalid) % death_invalid)
if place_references:
2009-02-03 10:33:19 +00:00
self.text.write(ngettext("%d place was referenced but not found\n", \
2009-01-31 11:29:44 +00:00
"%d places were referenced, but not found\n", place_references) % place_references)
if source_references:
2009-02-03 10:33:19 +00:00
self.text.write(ngettext("%d source was referenced but not found\n", \
2009-01-31 11:29:44 +00:00
"%d sources were referenced, but not found\n", source_references) % source_references)
if media_references:
2009-02-03 10:33:19 +00:00
self.text.write(ngettext("%d media object was referenced but not found\n", \
2009-01-31 11:29:44 +00:00
"%d media objects were referenced but not found\n", media_references) % media_references)
if note_references:
2009-02-03 10:33:19 +00:00
self.text.write(ngettext("%d note object was referenced but not found\n", \
2009-01-31 11:29:44 +00:00
"%d note objects were referenced but not found\n", note_references) % note_references)
if name_format:
2009-02-03 10:33:19 +00:00
self.text.write(ngettext("%d invalid name format reference was removed\n", \
2009-01-31 11:29:44 +00:00
"%d invalid name format references were removed\n", name_format) % name_format)
2003-05-19 04:54:14 +00:00
if empty_objs > 0 :
self.text.write(_("%d empty objects removed:\n"
" %d person objects\n"
" %d family objects\n"
" %d event objects\n"
" %d source objects\n"
" %d media objects\n"
" %d place objects\n"
" %d repository objects\n"
" %d note objects\n"
) % (empty_objs,
len(self.empty_objects['persons']),
len(self.empty_objects['families']),
len(self.empty_objects['events']),
len(self.empty_objects['sources']),
len(self.empty_objects['media']),
len(self.empty_objects['places']),
len(self.empty_objects['repos']),
len(self.empty_objects['notes'])
) )
2003-05-19 04:54:14 +00:00
return errors
2002-10-20 14:25:16 +00:00
#-------------------------------------------------------------------------
#
# Display the results
#
#-------------------------------------------------------------------------
class Report(ManagedWindow.ManagedWindow):
def __init__(self, uistate, text, cl=0):
if cl:
print text
return
ManagedWindow.ManagedWindow.__init__(self, uistate, [], self)
topDialog = Glade()
2009-08-15 14:37:14 +00:00
topDialog.get_object("close").connect('clicked', self.close)
window = topDialog.toplevel
textwindow = topDialog.get_object("textwindow")
textwindow.get_buffer().set_text(text)
self.set_window(window,
#topDialog.get_widget("title"),
topDialog.get_object("title"),
_("Integrity Check Results"))
self.show()
def build_menu_names(self, obj):
2009-08-15 14:37:14 +00:00
return (_('Check and Repair'), None)
2002-10-20 14:25:16 +00:00
2005-12-06 06:38:09 +00:00
#------------------------------------------------------------------------
#
#
#
#------------------------------------------------------------------------
class CheckOptions(Tool.ToolOptions):
"""
Defines options and provides handling interface.
"""
2009-08-15 14:37:14 +00:00
def __init__(self, name, person_id=None):
Tool.ToolOptions.__init__(self, name, person_id)