Commit c9ba36fb authored by Hanno Schlichting's avatar Hanno Schlichting

Store the value_indexes result in a module global. It is too expensive to calculate for each query

parent 095acc89
...@@ -18,9 +18,12 @@ from Acquisition import aq_base ...@@ -18,9 +18,12 @@ from Acquisition import aq_base
from Acquisition import aq_parent from Acquisition import aq_parent
from Products.PluginIndexes.interfaces import IUniqueValueIndex from Products.PluginIndexes.interfaces import IUniqueValueIndex
reportlock = allocate_lock() reports_lock = allocate_lock()
reports = {} reports = {}
value_indexes_lock = allocate_lock()
value_indexes = frozenset()
MAX_DISTINCT_VALUES = 10 MAX_DISTINCT_VALUES = 10
...@@ -33,15 +36,43 @@ def determine_value_indexes(catalog): ...@@ -33,15 +36,43 @@ def determine_value_indexes(catalog):
# of unique values, where the number of items for each value differs a # of unique values, where the number of items for each value differs a
# lot. If the number of items per value is similar, the duration of a # lot. If the number of items per value is similar, the duration of a
# query is likely similar as well. # query is likely similar as well.
valueindexes = [] global value_indexes
if value_indexes:
# Calculating all the value indexes is quite slow, so we do this once
# for the first query. Since this is an optimization only, slightly
# outdated results based on index changes in the running process
# can be ignored.
return value_indexes
new_value_indexes = set()
for name, index in catalog.indexes.items(): for name, index in catalog.indexes.items():
if IUniqueValueIndex.providedBy(index): if IUniqueValueIndex.providedBy(index):
values = index.uniqueValues() values = index.uniqueValues()
if values and len(values) < MAX_DISTINCT_VALUES: if values and len(values) < MAX_DISTINCT_VALUES:
# Only consider indexes which actually return a number # Only consider indexes which actually return a number
# greater than zero # greater than zero
valueindexes.append(name) new_value_indexes.add(name)
return frozenset(valueindexes) try:
value_indexes_lock.acquire()
value_indexes = frozenset(new_value_indexes)
finally:
value_indexes_lock.release()
return value_indexes
def clear_value_indexes():
global value_indexes
try:
value_indexes_lock.acquire()
value_indexes = frozenset()
finally:
value_indexes_lock.release()
from zope.testing.cleanup import addCleanUp
addCleanUp(clear_value_indexes)
del addCleanUp
def make_key(catalog, request): def make_key(catalog, request):
...@@ -142,7 +173,7 @@ class CatalogReport(StopWatch): ...@@ -142,7 +173,7 @@ class CatalogReport(StopWatch):
if res[0] < self.threshold: if res[0] < self.threshold:
return return
reportlock.acquire() reports_lock.acquire()
try: try:
if self.cid not in reports: if self.cid not in reports:
reports[self.cid] = {} reports[self.cid] = {}
...@@ -156,14 +187,14 @@ class CatalogReport(StopWatch): ...@@ -156,14 +187,14 @@ class CatalogReport(StopWatch):
reports[self.cid][key] = (1, res[0], res) reports[self.cid][key] = (1, res[0], res)
finally: finally:
reportlock.release() reports_lock.release()
def reset(self): def reset(self):
reportlock.acquire() reports_lock.acquire()
try: try:
reports[self.cid] = {} reports[self.cid] = {}
finally: finally:
reportlock.release() reports_lock.release()
def report(self): def report(self):
"""Returns a statistic report of catalog queries as list of dicts as """Returns a statistic report of catalog queries as list of dicts as
......
...@@ -855,8 +855,8 @@ class TestZCatalogGetObject(unittest.TestCase): ...@@ -855,8 +855,8 @@ class TestZCatalogGetObject(unittest.TestCase):
self.assertEqual(brain._unrestrictedGetObject(), None) self.assertEqual(brain._unrestrictedGetObject(), None)
class TestCatalogReport(unittest.TestCase): class TestCatalogReport(unittest.TestCase):
def setUp(self): def setUp(self):
from Products.ZCatalog.ZCatalog import ZCatalog from Products.ZCatalog.ZCatalog import ZCatalog
vocabulary = Vocabulary.Vocabulary( vocabulary = Vocabulary.Vocabulary(
...@@ -875,6 +875,10 @@ class TestCatalogReport(unittest.TestCase): ...@@ -875,6 +875,10 @@ class TestCatalogReport(unittest.TestCase):
obj.big = i > 5 obj.big = i > 5
self.zcat.catalog_object(obj, str(i)) self.zcat.catalog_object(obj, str(i))
def tearDown(self):
from Products.ZCatalog.CatalogReport import clear_value_indexes
clear_value_indexes()
def test_ReportLength(self): def test_ReportLength(self):
""" tests the report aggregation """ """ tests the report aggregation """
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment