back-port NickH's test fix from gramps40

Apply r22767, w/o the new test.

svn: r23172
This commit is contained in:
Vassilii Khachaturov 2013-09-20 12:01:00 +00:00
parent f2685cfc30
commit b01b42fab6
3 changed files with 118 additions and 284 deletions

View File

@ -1,26 +1,36 @@
#
# Gramps - a GTK+/GNOME based genealogy program
#
# Copyright (C) 2000-2007 Donald N. Allingham
#
# 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$
import unittest import unittest
import logging
import os import os
import tempfile import tempfile
import shutil import shutil
import time
import traceback
import sys
from bsddb import dbshelve, db from bsddb import dbshelve, db
sys.path.append('../../src')
try: try:
set() set()
except NameError: except NameError:
from sets import Set as set from sets import Set as set
import const
logger = logging.getLogger('Gramps.GrampsDbBase_Test')
from GrampsDbTestBase import GrampsDbBaseTest
class Data(object): class Data(object):
def __init__(self, handle,surname, name): def __init__(self, handle,surname, name):
@ -41,7 +51,21 @@ class CursorTest(unittest.TestCase):
self.env.set_cachesize(0,0x2000000) self.env.set_cachesize(0,0x2000000)
self.env.set_lk_max_locks(25000) self.env.set_lk_max_locks(25000)
self.env.set_lk_max_objects(25000) self.env.set_lk_max_objects(25000)
self.env.set_flags(db.DB_LOG_AUTOREMOVE,1) # clean up unused logs
# clean up unused logs
autoremove_flag = None
autoremove_method = None
for flag in ["DB_LOG_AUTO_REMOVE", "DB_LOG_AUTOREMOVE"]:
if hasattr(db, flag):
autoremove_flag = getattr(db, flag)
break
for method in ["log_set_config", "set_flags"]:
if hasattr(self.env, method):
autoremove_method = getattr(self.env, method)
break
if autoremove_method and autoremove_flag:
autoremove_method(autoremove_flag, 1)
# The DB_PRIVATE flag must go if we ever move to multi-user setup # The DB_PRIVATE flag must go if we ever move to multi-user setup
env_flags = db.DB_CREATE|db.DB_RECOVER|db.DB_PRIVATE|\ env_flags = db.DB_CREATE|db.DB_RECOVER|db.DB_PRIVATE|\
db.DB_INIT_MPOOL|db.DB_INIT_LOCK|\ db.DB_INIT_MPOOL|db.DB_INIT_LOCK|\
@ -82,12 +106,12 @@ class CursorTest(unittest.TestCase):
data = Data(str(1),'surname1','name1') data = Data(str(1),'surname1','name1')
the_txn = self.env.txn_begin() the_txn = self.env.txn_begin()
self.person_map.put(data.handle,data,txn=the_txn) self.person_map.put(data.handle, data, txn=the_txn)
the_txn.commit() the_txn.commit()
v = self.person_map.get(data.handle) v = self.person_map.get(data.handle)
assert v.handle == data.handle self.assertEqual(v.handle, data.handle)
def test_insert_with_curor_closed(self): def test_insert_with_curor_closed(self):
"""test_insert_with_curor_closed""" """test_insert_with_curor_closed"""
@ -100,15 +124,16 @@ class CursorTest(unittest.TestCase):
cursor.close() cursor.close()
cursor_txn.commit() cursor_txn.commit()
data = Data(str(2),'surname2','name2') data = Data(str(2), 'surname2', 'name2')
the_txn = self.env.txn_begin() the_txn = self.env.txn_begin()
self.person_map.put(data.handle,data,txn=the_txn) self.person_map.put(data.handle, data, txn=the_txn)
the_txn.commit() the_txn.commit()
v = self.person_map.get(data.handle) v = self.person_map.get(data.handle)
assert v.handle == data.handle self.assertEqual(v.handle, data.handle)
@unittest.skip("Insert expected to fail with open cursor")
def test_insert_with_curor_open(self): def test_insert_with_curor_open(self):
"""test_insert_with_curor_open""" """test_insert_with_curor_open"""
@ -117,9 +142,9 @@ class CursorTest(unittest.TestCase):
cursor.first() cursor.first()
cursor.next() cursor.next()
data = Data(str(2),'surname2','name2') data = Data(str(2),'surname2', 'name2')
the_txn = self.env.txn_begin() the_txn = self.env.txn_begin()
self.person_map.put(data.handle,data,txn=the_txn) self.person_map.put(data.handle, data, txn=the_txn)
the_txn.commit() the_txn.commit()
cursor.close() cursor.close()
@ -127,9 +152,10 @@ class CursorTest(unittest.TestCase):
v = self.person_map.get(data.handle) v = self.person_map.get(data.handle)
assert v.handle == data.handle self.assertEqual(v.handle, data.handle)
def xtest_insert_with_curor_open_and_db_open(self): @unittest.skip("Insert expected to fail with open cursor")
def test_insert_with_curor_open_and_db_open(self):
"""test_insert_with_curor_open_and_db_open""" """test_insert_with_curor_open_and_db_open"""
(person2,surnames2) = self._open_tables() (person2,surnames2) = self._open_tables()
@ -139,9 +165,9 @@ class CursorTest(unittest.TestCase):
cursor.first() cursor.first()
cursor.next() cursor.next()
data = Data(str(2),'surname2','name2') data = Data(str(2),'surname2', 'name2')
the_txn = self.env.txn_begin() the_txn = self.env.txn_begin()
self.person_map.put(data.handle,data,txn=the_txn) self.person_map.put(data.handle, data, txn=the_txn)
the_txn.commit() the_txn.commit()
cursor.close() cursor.close()
@ -149,11 +175,11 @@ class CursorTest(unittest.TestCase):
v = self.person_map.get(data.handle) v = self.person_map.get(data.handle)
assert v.handle == data.handle self.assertEqual(v.handle, data.handle)
def testSuite(): def testSuite():
suite = unittest.makeSuite(CursorTest,'test') suite = unittest.makeSuite(CursorTest, 'test')
return suite return suite

View File

@ -1,25 +1,38 @@
#
# Gramps - a GTK+/GNOME based genealogy program
#
# Copyright (C) 2000-2007 Donald N. Allingham
#
# 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$
import unittest import unittest
import logging
import os
import tempfile import tempfile
import shutil import shutil
import time
import traceback
import sys
sys.path.append('../src')
try: try:
set() set()
except NameError: except NameError:
from sets import Set as set from sets import Set as set
from gen.db import DbBsddb
from cli.clidbman import CLIDbManager
import const
import gen.lib
logger = logging.getLogger('Gramps.GrampsDbTestBase') from gramps.gen.db import DbBsddb, DbTxn
from gramps.cli.clidbman import CLIDbManager
from gramps.gen.lib import (Source, RepoRef, Citation, Repository, Person,
Family, Event, Place, MediaObject)
class GrampsDbBaseTest(unittest.TestCase): class GrampsDbBaseTest(unittest.TestCase):
"""Base class for unittest that need to be able to create """Base class for unittest that need to be able to create
@ -29,7 +42,6 @@ class GrampsDbBaseTest(unittest.TestCase):
def dummy_callback(dummy): def dummy_callback(dummy):
pass pass
self._tmpdir = tempfile.mkdtemp() self._tmpdir = tempfile.mkdtemp()
#self._filename = os.path.join(self._tmpdir,'test.grdb')
self._db = DbBsddb() self._db = DbBsddb()
dbman = CLIDbManager(None) dbman = CLIDbManager(None)
@ -84,79 +96,77 @@ class GrampsDbBaseTest(unittest.TestCase):
def _add_source(self,repos=None): def _add_source(self,repos=None):
# Add a Source # Add a Source
tran = self._db.transaction_begin()
source = gen.lib.Source()
if repos is not None:
repo_ref = gen.lib.RepoRef()
repo_ref.set_reference_handle(repos.get_handle())
source.add_repo_reference(repo_ref)
self._db.add_source(source,tran)
self._db.commit_source(source,tran)
self._db.transaction_commit(tran, "Add Source")
return source with DbTxn("Add Source and Citation", self._db) as tran:
source = Source()
if repos is not None:
repo_ref = RepoRef()
repo_ref.set_reference_handle(repos.get_handle())
source.add_repo_reference(repo_ref)
self._db.add_source(source, tran)
self._db.commit_source(source, tran)
citation = Citation()
citation.set_reference_handle(source.get_handle())
self._db.add_citation(citation, tran)
self._db.commit_citation(citation, tran)
return citation
def _add_repository(self): def _add_repository(self):
# Add a Repository # Add a Repository
tran = self._db.transaction_begin() with DbTxn("Add Repository", self._db) as tran:
repos = gen.lib.Repository() repos = Repository()
self._db.add_repository(repos,tran) self._db.add_repository(repos, tran)
self._db.commit_repository(repos,tran) self._db.commit_repository(repos, tran)
self._db.transaction_commit(tran, "Add Repository")
return repos return repos
def _add_object_with_source(self,sources, object_class,add_method,commit_method): def _add_object_with_source(self, citations, object_class, add_method,
commit_method):
object = object_class() object = object_class()
for source in sources: with DbTxn("Add Object", self._db) as tran:
src_ref = gen.lib.SourceRef() for citation in citations:
src_ref.set_reference_handle(source.get_handle()) object.add_citation(citation.get_handle())
object.add_source_reference(src_ref) add_method(object, tran)
commit_method(object, tran)
tran = self._db.transaction_begin()
add_method(object,tran)
commit_method(object,tran)
self._db.transaction_commit(tran, "Add Object")
return object return object
def _add_person_with_sources(self,sources): def _add_person_with_sources(self, citations):
return self._add_object_with_source(sources, return self._add_object_with_source(citations,
gen.lib.Person, Person,
self._db.add_person, self._db.add_person,
self._db.commit_person) self._db.commit_person)
def _add_family_with_sources(self,sources): def _add_family_with_sources(self, citations):
return self._add_object_with_source(sources, return self._add_object_with_source(citations,
gen.lib.Family, Family,
self._db.add_family, self._db.add_family,
self._db.commit_family) self._db.commit_family)
def _add_event_with_sources(self,sources): def _add_event_with_sources(self, citations):
return self._add_object_with_source(sources, return self._add_object_with_source(citations,
gen.lib.Event, Event,
self._db.add_event, self._db.add_event,
self._db.commit_event) self._db.commit_event)
def _add_place_with_sources(self,sources): def _add_place_with_sources(self, citations):
return self._add_object_with_source(sources, return self._add_object_with_source(citations,
gen.lib.Place, Place,
self._db.add_place, self._db.add_place,
self._db.commit_place) self._db.commit_place)
def _add_media_object_with_sources(self,sources): def _add_media_object_with_sources(self, citations):
return self._add_object_with_source(sources, return self._add_object_with_source(citations,
gen.lib.MediaObject, MediaObject,
self._db.add_object, self._db.add_object,
self._db.commit_media_object) self._db.commit_media_object)

View File

@ -1,202 +0,0 @@
import unittest
import logging
import os
import tempfile
import shutil
import time
import traceback
import sys
sys.path.append('../../src')
try:
set()
except NameError:
from sets import Set as set
import const
import gen.lib
logger = logging.getLogger('Gramps.GrampsDbBase_Test')
from GrampsDbTestBase import GrampsDbBaseTest
class ReferenceMapTest (GrampsDbBaseTest):
"""Test methods on the GrampsDbBase class that are related to the reference_map
index implementation."""
def test_simple_lookup(self):
"""insert a record and a reference and check that
a lookup for the reference returns the original
record."""
source = self._add_source()
person = self._add_person_with_sources([source])
references = list(self._db.find_backlink_handles(source.get_handle()))
assert len(references) == 1
assert references[0] == (gen.lib.Person.__name__,person.get_handle())
def test_backlink_for_repository(self):
"""check that the source / repos backlink lookup works."""
repos = self._add_repository()
source = self._add_source(repos=repos)
references = list(self._db.find_backlink_handles(repos.get_handle()))
assert len(references) == 1
assert references[0] == (gen.lib.Source.__name__,source.get_handle())
def test_class_limited_lookup(self):
"""check that class limited lookups work."""
source = self._add_source()
person = self._add_person_with_sources([source])
self._add_family_with_sources([source])
self._add_event_with_sources([source])
self._add_place_with_sources([source])
self._add_media_object_with_sources([source])
# make sure that we have the correct number of references (one for each object)
references = list(self._db.find_backlink_handles(source.get_handle()))
assert len(references) == 5, "len(references) == %s " % str(len(references))
# should just return the person reference
references = [ ref for ref in self._db.find_backlink_handles(source.get_handle(),(gen.lib.Person.__name__,)) ]
assert len(references) == 1, "len(references) == %s " % str(len(references))
assert references[0][0] == gen.lib.Person.__name__, "references = %s" % repr(references)
# should just return the person and event reference
references = list(self._db.find_backlink_handles(source.get_handle(),
(gen.lib.Person.__name__, gen.lib.Event.__name__)))
assert len(references) == 2, "len(references) == %s " % str(len(references))
assert references[0][0] == gen.lib.Person.__name__, "references = %s" % repr(references)
assert references[1][0] == gen.lib.Event.__name__, "references = %s" % repr(references)
def test_delete_primary(self):
"""check that deleting a primary will remove the backreferences
from the reference_map"""
source = self._add_source()
person = self._add_person_with_sources([source])
assert self._db.get_person_from_handle(person.get_handle()) is not None
tran = self._db.transaction_begin()
self._db.remove_person(person.get_handle(),tran)
self._db.transaction_commit(tran, "Del Person")
assert self._db.get_person_from_handle(person.get_handle()) is None
references = list(self._db.find_backlink_handles(source.get_handle()))
assert len(references) == 0, "len(references) == %s " % str(len(references))
def test_reindex_reference_map(self):
"""Test that the reindex function works."""
def cb(count):
pass
# unhook the reference_map update function so that we
# can insert some records without the reference_map being updated.
update_method = self._db.update_reference_map
self._db._update_reference_map = lambda x,y: 1
# Insert a person/source pair.
source = self._add_source()
person = self._add_person_with_sources([source])
# Check that the reference map does not contain the reference.
references = list(self._db.find_backlink_handles(source.get_handle()))
assert len(references) == 0, "len(references) == %s " % str(len(references))
# Reinstate the reference_map method and reindex the database
self._db._update_reference_map = update_method
self._db.reindex_reference_map(cb)
# Check that the reference now appears in the reference_map
references = list(self._db.find_backlink_handles(source.get_handle()))
assert len(references) == 1, "len(references) == %s " % str(len(references))
def perf_simple_search_speed(self):
num_sources = 100
num_persons = 1000
num_families = 10
num_events = 10
num_places = 10
num_media_objects = 10
num_links = 10
self._populate_database(num_sources,
num_persons,
num_families,
num_events,
num_places,
num_media_objects,
num_links)
# time searching for source backrefs with and without reference_map
cur = self._db.get_source_cursor()
handle,data = cur.first()
cur.close()
start = time.time()
references = list(self._db.find_backlink_handles(handle))
end = time.time()
with_reference_map = end - start
remember = self._db.__class__.find_backlink_handles
self._db.__class__.find_backlink_handles = self._db.__class__.__base__.find_backlink_handles
start = time.time()
references = list(self._db.find_backlink_handles(handle))
end = time.time()
without_reference_map = end - start
self._db.__class__.find_backlink_handles = remember
logger.info("search test with following data: \n"
"num_sources = %d \n"
"num_persons = %d \n"
"num_families = %d \n"
"num_events = %d \n"
"num_places = %d \n"
"num_media_objects = %d \n"
"num_links = %d" % (num_sources,
num_persons,
num_families,
num_events,
num_places,
num_media_objects,
num_links))
logger.info("with refs %s\n", str(with_reference_map))
logger.info("without refs %s\n", str(without_reference_map))
assert with_reference_map < (without_reference_map / 10), "Reference_map should an order of magnitude faster."
def testSuite():
suite = unittest.makeSuite(ReferenceMapTest,'test')
return suite
def perfSuite():
return unittest.makeSuite(ReferenceMapTest,'perf')
if __name__ == '__main__':
unittest.TextTestRunner().run(testSuite())