2006-03-18 01:30:23 +00:00
|
|
|
#
|
|
|
|
# Gramps - a GTK+/GNOME based genealogy program
|
|
|
|
#
|
2006-04-13 03:15:22 +00:00
|
|
|
# Copyright (C) 2004-2006 Donald N. Allingham
|
2006-03-18 01:30:23 +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: __init__.py 6086 2006-03-06 03:54:58Z dallingham $
|
|
|
|
|
2006-05-18 03:14:55 +00:00
|
|
|
from gettext import gettext as _
|
2006-03-18 01:30:23 +00:00
|
|
|
|
2006-04-09 22:53:53 +00:00
|
|
|
import RelLib
|
2006-05-15 21:56:57 +00:00
|
|
|
from BasicUtils import UpdateCallback
|
2006-04-09 22:53:53 +00:00
|
|
|
|
2006-03-18 01:30:23 +00:00
|
|
|
def remove_family_relationships(db, family_handle, trans=None):
|
2006-03-18 03:21:03 +00:00
|
|
|
family = db.get_family_from_handle(family_handle)
|
2006-03-18 01:30:23 +00:00
|
|
|
|
|
|
|
if trans == None:
|
|
|
|
need_commit = True
|
|
|
|
trans = db.transaction_begin()
|
|
|
|
else:
|
|
|
|
need_commit = False
|
|
|
|
|
|
|
|
for phandle in [ family.get_father_handle(),
|
|
|
|
family.get_mother_handle()]:
|
|
|
|
if phandle:
|
|
|
|
person = db.get_person_from_handle(phandle)
|
2006-03-18 03:21:03 +00:00
|
|
|
person.remove_family_handle(family_handle)
|
|
|
|
db.commit_person(person, trans)
|
2006-03-18 01:30:23 +00:00
|
|
|
|
2006-04-21 20:39:30 +00:00
|
|
|
for ref in family.get_child_ref_list():
|
|
|
|
phandle = ref.ref
|
2006-03-18 01:30:23 +00:00
|
|
|
person = db.get_person_from_handle(phandle)
|
2006-03-18 03:21:03 +00:00
|
|
|
person.remove_parent_family_handle(family_handle)
|
|
|
|
db.commit_person(person, trans)
|
2006-03-18 01:30:23 +00:00
|
|
|
|
2006-03-18 03:21:03 +00:00
|
|
|
db.remove_family(family_handle, trans)
|
2006-03-18 01:30:23 +00:00
|
|
|
|
|
|
|
if need_commit:
|
2006-03-18 03:21:03 +00:00
|
|
|
db.transaction_commit(trans, _("Remove Family"))
|
2006-03-18 01:30:23 +00:00
|
|
|
|
|
|
|
def remove_parent_from_family(db, person_handle, family_handle, trans=None):
|
|
|
|
"""
|
|
|
|
Removes a person as either the father or mother of a family,
|
|
|
|
deleting the family if it becomes empty.
|
|
|
|
"""
|
|
|
|
person = db.get_person_from_handle(person_handle)
|
|
|
|
family = db.get_family_from_handle(family_handle)
|
|
|
|
|
|
|
|
if trans == None:
|
|
|
|
need_commit = True
|
|
|
|
trans = db.transaction_begin()
|
|
|
|
else:
|
|
|
|
need_commit = False
|
|
|
|
|
|
|
|
person.remove_family_handle(family_handle)
|
|
|
|
if family.get_father_handle() == person_handle:
|
|
|
|
family.set_father_handle(None)
|
|
|
|
elif family.get_mother_handle() == person_handle:
|
|
|
|
family.set_mother_handle(None)
|
|
|
|
|
2006-04-20 03:50:57 +00:00
|
|
|
child_list = family.get_child_ref_list()
|
2006-03-18 01:30:23 +00:00
|
|
|
if (not family.get_father_handle() and not family.get_mother_handle() and
|
2006-04-12 03:06:25 +00:00
|
|
|
len(child_list) <= 1):
|
2006-03-18 01:30:23 +00:00
|
|
|
db.remove_family(family_handle, trans)
|
2006-04-12 03:06:25 +00:00
|
|
|
if child_list:
|
2006-04-20 03:50:57 +00:00
|
|
|
child = db.get_person_from_handle(child_list[0].ref)
|
2006-04-12 03:06:25 +00:00
|
|
|
child.remove_parent_family_handle(family_handle)
|
|
|
|
db.commit_person(child, trans)
|
2006-03-18 01:30:23 +00:00
|
|
|
msg = _("Remove father from family")
|
|
|
|
else:
|
|
|
|
db.commit_family(family, trans)
|
|
|
|
msg = _("Remove mother from family")
|
|
|
|
db.commit_person(person, trans)
|
|
|
|
|
|
|
|
if need_commit:
|
|
|
|
db.transaction_commit(trans,msg)
|
|
|
|
|
|
|
|
def remove_child_from_family(db, person_handle, family_handle, trans=None):
|
|
|
|
"""
|
|
|
|
Removes a person as a child of the family, deleting the family if
|
|
|
|
it becomes empty.
|
|
|
|
"""
|
|
|
|
person = db.get_person_from_handle(person_handle)
|
|
|
|
family = db.get_family_from_handle(family_handle)
|
|
|
|
person.remove_parent_family_handle(family_handle)
|
|
|
|
family.remove_child_handle(person_handle)
|
|
|
|
|
|
|
|
if trans == None:
|
|
|
|
need_commit = True
|
|
|
|
trans = db.transaction_begin()
|
|
|
|
else:
|
|
|
|
need_commit = False
|
|
|
|
|
2006-04-20 03:50:57 +00:00
|
|
|
child_list = family.get_child_ref_list()
|
2006-03-18 01:30:23 +00:00
|
|
|
if (not family.get_father_handle() and not family.get_mother_handle() and
|
2006-04-12 03:10:28 +00:00
|
|
|
len(child_list) <= 1):
|
2006-03-18 01:30:23 +00:00
|
|
|
db.remove_family(family_handle, trans)
|
2006-04-12 03:10:28 +00:00
|
|
|
if child_list:
|
2006-04-20 03:50:57 +00:00
|
|
|
child = db.get_person_from_handle(child_list[0].ref)
|
2006-04-12 03:10:28 +00:00
|
|
|
child.remove_parent_family_handle(family_handle)
|
|
|
|
db.commit_person(child, trans)
|
2006-03-18 01:30:23 +00:00
|
|
|
else:
|
|
|
|
db.commit_family(family, trans)
|
|
|
|
db.commit_person(person, trans)
|
|
|
|
|
|
|
|
if need_commit:
|
|
|
|
db.transaction_commit(trans,_("Remove child from family"))
|
2006-04-09 22:53:53 +00:00
|
|
|
|
|
|
|
|
2006-04-13 03:15:22 +00:00
|
|
|
def add_child_to_family(db, family, child,
|
2006-04-14 04:36:25 +00:00
|
|
|
mrel=RelLib.ChildRefType(),
|
|
|
|
frel=RelLib.ChildRefType(),
|
2006-04-13 03:15:22 +00:00
|
|
|
trans=None):
|
2006-04-09 22:53:53 +00:00
|
|
|
|
2006-04-24 04:06:17 +00:00
|
|
|
cref = RelLib.ChildRef()
|
|
|
|
cref.ref = child.handle
|
|
|
|
cref.set_father_relation(frel)
|
|
|
|
cref.set_mother_relation(mrel)
|
|
|
|
|
|
|
|
family.add_child_ref(cref)
|
|
|
|
child.add_parent_family_handle(family.handle)
|
2006-04-09 22:53:53 +00:00
|
|
|
|
|
|
|
if trans == None:
|
|
|
|
need_commit = True
|
|
|
|
trans = db.transaction_begin()
|
|
|
|
else:
|
|
|
|
need_commit = False
|
|
|
|
|
|
|
|
db.commit_family(family,trans)
|
2006-04-24 04:06:17 +00:00
|
|
|
db.commit_person(child,trans)
|
2006-04-09 22:53:53 +00:00
|
|
|
|
|
|
|
if need_commit:
|
|
|
|
db.transaction_commit(trans, _('Add child to family') )
|
2006-05-10 05:52:55 +00:00
|
|
|
|
|
|
|
|
2006-05-11 06:16:45 +00:00
|
|
|
def get_total(db):
|
|
|
|
person_len = db.get_number_of_people()
|
|
|
|
family_len = db.get_number_of_families()
|
|
|
|
event_len = db.get_number_of_events()
|
|
|
|
source_len = db.get_number_of_sources()
|
|
|
|
place_len = db.get_number_of_places()
|
|
|
|
repo_len = db.get_number_of_repositories()
|
|
|
|
obj_len = db.get_number_of_media_objects()
|
2006-05-11 05:11:47 +00:00
|
|
|
|
2006-05-11 06:16:45 +00:00
|
|
|
return person_len + family_len + event_len + \
|
|
|
|
place_len + source_len + obj_len + repo_len
|
2006-05-10 05:52:55 +00:00
|
|
|
|
2006-05-11 05:11:47 +00:00
|
|
|
def db_copy(from_db,to_db,callback):
|
2006-05-20 00:52:47 +00:00
|
|
|
"""
|
|
|
|
Copy all data in from_db into to_db.
|
|
|
|
|
|
|
|
Both databases must be loaded.
|
|
|
|
It is assumed that to_db is an empty database,
|
|
|
|
so no care is taken to prevent handle collision or merge data.
|
|
|
|
"""
|
|
|
|
|
2006-05-16 03:47:58 +00:00
|
|
|
uc = UpdateCallback(callback)
|
2006-05-11 06:16:45 +00:00
|
|
|
uc.set_total(get_total(from_db))
|
2006-05-11 05:11:47 +00:00
|
|
|
|
2006-05-16 03:47:58 +00:00
|
|
|
tables = {
|
2006-05-10 05:52:55 +00:00
|
|
|
'Person': {'cursor_func': from_db.get_person_cursor,
|
2006-05-16 03:47:58 +00:00
|
|
|
'table': to_db.person_map,
|
|
|
|
'sec_table' : to_db.id_trans },
|
2006-05-10 05:52:55 +00:00
|
|
|
'Family': {'cursor_func': from_db.get_family_cursor,
|
2006-05-16 03:47:58 +00:00
|
|
|
'table': to_db.family_map,
|
|
|
|
'sec_table' : to_db.fid_trans },
|
2006-05-10 05:52:55 +00:00
|
|
|
'Event': {'cursor_func': from_db.get_event_cursor,
|
2006-05-16 03:47:58 +00:00
|
|
|
'table': to_db.event_map,
|
|
|
|
'sec_table' : to_db.eid_trans },
|
2006-05-10 05:52:55 +00:00
|
|
|
'Place': {'cursor_func': from_db.get_place_cursor,
|
2006-05-16 03:47:58 +00:00
|
|
|
'table': to_db.place_map,
|
|
|
|
'sec_table' : to_db.pid_trans },
|
2006-05-10 05:52:55 +00:00
|
|
|
'Source': {'cursor_func': from_db.get_source_cursor,
|
2006-05-16 03:47:58 +00:00
|
|
|
'table': to_db.source_map,
|
|
|
|
'sec_table' : to_db.sid_trans },
|
2006-05-10 05:52:55 +00:00
|
|
|
'MediaObject': {'cursor_func': from_db.get_media_cursor,
|
2006-05-16 03:47:58 +00:00
|
|
|
'table': to_db.media_map,
|
|
|
|
'sec_table' : to_db.oid_trans },
|
2006-05-10 05:52:55 +00:00
|
|
|
'Repository': {'cursor_func': from_db.get_repository_cursor,
|
2006-05-16 03:47:58 +00:00
|
|
|
'table': to_db.repository_map,
|
|
|
|
'sec_table' : to_db.rid_trans },
|
2006-05-10 05:52:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if to_db.__class__.__name__ == 'GrampsBSDDB':
|
|
|
|
if to_db.UseTXN:
|
|
|
|
add_data = add_data_txn
|
|
|
|
else:
|
|
|
|
add_data = add_data_notxn
|
2006-05-16 03:47:58 +00:00
|
|
|
update_secondary = update_secondary_empty
|
2006-05-10 05:52:55 +00:00
|
|
|
else:
|
|
|
|
add_data = add_data_dict
|
2006-05-16 03:47:58 +00:00
|
|
|
# For InMem databases, the secondary indices need to be
|
|
|
|
# created as we copy objects
|
|
|
|
update_secondary = update_secondary_inmem
|
2006-05-10 05:52:55 +00:00
|
|
|
|
2006-05-10 14:23:38 +00:00
|
|
|
# Start batch transaction to use async TXN and other tricks
|
|
|
|
trans = to_db.transaction_begin("",batch=True)
|
|
|
|
|
2006-05-16 03:47:58 +00:00
|
|
|
for table_name in tables.keys():
|
|
|
|
cursor_func = tables[table_name]['cursor_func']
|
|
|
|
table = tables[table_name]['table']
|
|
|
|
sec_table = tables[table_name]['sec_table']
|
2006-05-10 05:52:55 +00:00
|
|
|
|
|
|
|
cursor = cursor_func()
|
|
|
|
item = cursor.first()
|
|
|
|
while item:
|
|
|
|
(handle,data) = item
|
|
|
|
add_data(to_db,table,handle,data)
|
2006-05-16 03:47:58 +00:00
|
|
|
update_secondary(sec_table,handle,data)
|
2006-05-10 05:52:55 +00:00
|
|
|
item = cursor.next()
|
2006-05-11 05:11:47 +00:00
|
|
|
uc.update()
|
2006-05-10 05:52:55 +00:00
|
|
|
cursor.close()
|
|
|
|
|
2006-05-10 14:23:38 +00:00
|
|
|
# Commit batch transaction: does nothing, except undoing the tricks
|
|
|
|
to_db.transaction_commit(trans,"")
|
|
|
|
|
2006-05-18 22:56:13 +00:00
|
|
|
# Copy bookmarks over:
|
|
|
|
# we already know that there's no overlap in handles anywhere
|
|
|
|
to_db.bookmarks = from_db.bookmarks
|
|
|
|
to_db.family_bookmarks = from_db.family_bookmarks
|
|
|
|
to_db.event_bookmarks = from_db.event_bookmarks
|
|
|
|
to_db.source_bookmarks = from_db.source_bookmarks
|
|
|
|
to_db.place_bookmarks = from_db.place_bookmarks
|
|
|
|
to_db.media_bookmarks = from_db.media_bookmarks
|
|
|
|
to_db.repo_bookmarks = from_db.repo_bookmarks
|
|
|
|
|
|
|
|
# Copy gender stats
|
|
|
|
to_db.genderStats = from_db.genderStats
|
|
|
|
|
2006-05-20 00:52:47 +00:00
|
|
|
# Copy custom types
|
|
|
|
to_db.family_event_names = from_db.family_event_names
|
|
|
|
to_db.individual_event_names = from_db.individual_event_names
|
|
|
|
to_db.family_attributes = from_db.family_attributes
|
|
|
|
to_db.individual_attributes = from_db.individual_attributes
|
|
|
|
to_db.marker_names = from_db.marker_names
|
|
|
|
to_db.child_ref_types = from_db.child_ref_types
|
|
|
|
to_db.family_rel_types = from_db.family_rel_types
|
|
|
|
to_db.event_role_names = from_db.event_role_names
|
|
|
|
to_db.name_types = from_db.name_types
|
|
|
|
to_db.repository_types = from_db.repository_types
|
|
|
|
to_db.source_media_types = from_db.source_media_types
|
|
|
|
to_db.url_types = from_db.url_types
|
|
|
|
to_db.media_attributes = from_db.media_attributes
|
2006-05-16 03:47:58 +00:00
|
|
|
|
2006-05-10 05:52:55 +00:00
|
|
|
def add_data_txn(db,table,handle,data):
|
|
|
|
the_txn = db.env.txn_begin()
|
|
|
|
table.put(handle,data,txn=the_txn)
|
|
|
|
the_txn.commit()
|
|
|
|
|
|
|
|
def add_data_notxn(db,table,handle,data):
|
|
|
|
table.put(handle,data)
|
|
|
|
|
|
|
|
def add_data_dict(db,table,handle,data):
|
|
|
|
table[handle] = data
|
2006-05-16 03:47:58 +00:00
|
|
|
|
|
|
|
def update_secondary_empty(sec_table,handle,data):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def update_secondary_inmem(sec_table,handle,data):
|
|
|
|
sec_table[str(data[1])] = str(handle)
|
2006-05-23 21:07:26 +00:00
|
|
|
|
|
|
|
def set_birth_death_index(db, person):
|
2006-06-06 18:49:34 +00:00
|
|
|
birth_ref_index = -1
|
|
|
|
death_ref_index = -1
|
|
|
|
event_ref_list = person.get_event_ref_list()
|
|
|
|
for index in range(len(event_ref_list)):
|
|
|
|
ref = event_ref_list[index]
|
2006-05-23 21:07:26 +00:00
|
|
|
event = db.get_event_from_handle(ref.ref)
|
2006-06-06 18:49:34 +00:00
|
|
|
if (int(event.get_type()) == RelLib.EventType.BIRTH) \
|
|
|
|
and (int(ref.get_role()) == RelLib.EventRoleType.PRIMARY) \
|
|
|
|
and (birth_ref_index == -1):
|
|
|
|
birth_ref_index = index
|
|
|
|
elif (int(event.get_type()) == RelLib.EventType.DEATH) \
|
|
|
|
and (int(ref.get_role()) == RelLib.EventRoleType.PRIMARY) \
|
|
|
|
and (death_ref_index == -1):
|
|
|
|
death_ref_index = index
|
|
|
|
|
|
|
|
person.birth_ref_index = birth_ref_index
|
|
|
|
person.death_ref_index = death_ref_index
|