############################################################################## # # Copyright (c) 2002-2006 Nexedi SARL and Contributors. All Rights Reserved. # Copyright (c) 2007-2009 Nexedi SA and Contributors. All Rights Reserved. # Jean-Paul Smets-Solanes <jp@nexedi.com> # Vincent Pelletier <vincent@nexedi.com> # # WARNING: This program as such is intended to be used by professional # programmers who take the whole responsability of assessing all potential # consequences resulting from its eventual inadequacies and bugs # End users who are looking for a ready-to-use solution with commercial # garantees and support are strongly adviced to contract a Free Software # Service Company # # 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. # ############################################################################## from Products.ZSQLCatalog.SQLExpression import SQLExpression from Products.ZSQLCatalog.ColumnMap import ColumnMap from zLOG import LOG from Products.ZSQLCatalog.interfaces.entire_query import IEntireQuery from zope.interface.verify import verifyClass from zope.interface import implements from Products.ZSQLCatalog.SQLCatalog import profiler_decorator def defaultDict(value): if value is None: return {} assert isinstance(value, dict) return value class EntireQuery(object): """ This is not a Query subclass, since it does not define a registerColumnMap method, and instead does the ColumnMap handling internaly. """ implements(IEntireQuery) column_map = None @profiler_decorator def __init__(self, query, order_by_list=(), group_by_list=(), select_dict=None, limit=None, catalog_table_name=None, extra_column_list=(), from_expression=None, order_by_override_list=None): self.query = query self.order_by_list = list(order_by_list) self.order_by_override_set = frozenset(order_by_override_list) self.group_by_list = list(group_by_list) self.select_dict = defaultDict(select_dict) self.limit = limit self.catalog_table_name = catalog_table_name self.extra_column_list = list(extra_column_list) self.from_expression = from_expression def asSearchTextExpression(self, sql_catalog): return self.query.asSearchTextExpression(sql_catalog) @profiler_decorator def asSQLExpression(self, sql_catalog, only_group_columns): column_map = self.column_map if column_map is None: # XXX: should we provide a way to register column map as a separate # method or do it here ? # Column Map was not built yet, do it. self.column_map = column_map = ColumnMap(catalog_table_name=self.catalog_table_name) for extra_column in self.extra_column_list: table, column = extra_column.replace('`', '').split('.') if table != self.catalog_table_name: raise ValueError, 'Extra columns must be catalog columns. %r does not follow this rule (catalog=%r, extra_column_list=%r)' % (extra_column, self.catalog_table_name, self.extra_column_list) column_map.registerColumn(extra_column) for column in self.group_by_list: column_map.registerColumn(column) for alias, column in self.select_dict.iteritems(): if column is None: column = alias else: column_map.ignoreColumn(alias) column_map.registerColumn(column) for override in self.order_by_override_set: column_map.ignoreColumn(override) for order_by in self.order_by_list: assert isinstance(order_by, (tuple, list)) assert len(order_by) column_map.registerColumn(order_by[0]) self.query.registerColumnMap(sql_catalog, column_map) column_map.build(sql_catalog) # Replace given group_by_list entries by their mapped representations. new_column_list = [] append = new_column_list.append for column in self.group_by_list: try: append(column_map.asSQLColumn(column)) except KeyError: LOG('EntireQuery', 100, 'Group-by column %r could not be mapped, but is passed through. This use is strongly discouraged.' % (column, )) append(column) self.group_by_list = new_column_list # Build a dictionnary from select_dict aliasing their mapped representations self.final_select_dict = select_dict = {} for alias, raw_column in self.select_dict.iteritems(): if raw_column is None: column = alias else: column = raw_column try: rendered = column_map.asSQLColumn(column) except KeyError: LOG('EntireQuery', 100, 'Select column %r could not be mapped, but is passed through. This use is strongly discouraged.' % (column, )) rendered = column select_dict[alias] = rendered # Replace given order_by_list entries by their mapped representations. new_order_by_list = [] append = new_order_by_list.append for order_by in self.order_by_list: column = order_by[0] if column in self.order_by_override_set: LOG('EntireQuery', 100, 'Order-by column %r is forcibly accepted. This use is strongly discouraged.' % (column, )) rendered = column else: try: rendered = column_map.asSQLColumn(column) except KeyError: LOG('SQLCatalog', 100, 'Order by %r ignored: it could not be mapped to a known column.' % (order_by, )) rendered = None if rendered is not None: if len(order_by) > 1: if len(order_by) > 2 and order_by[2] not in (None, ''): rendered = 'CAST(%s AS %s)' % (rendered, order_by[2]) rendered = '%s %s' % (rendered, order_by[1]) append(rendered) self.order_by_list = new_order_by_list # generate SQLExpression from query sql_expression_list = [self.query.asSQLExpression(sql_catalog, column_map, only_group_columns)] # generate join expression based on column_map.getJoinTableAliasList append = sql_expression_list.append for join_query in column_map.iterJoinQueryList(): append(join_query.asSQLExpression(sql_catalog, column_map, only_group_columns)) join_table_list = column_map.getJoinTableAliasList() if len(join_table_list): # XXX: Is there any special rule to observe when joining tables ? # Maybe we could check which column is a primary key instead of # hardcoding "uid". where_pattern = '`%s`.`uid` = `%%s`.`uid`' % \ (column_map.getCatalogTableAlias(), ) # XXX: It would cleaner from completeness point of view to use column # mapper to render column, but makes code much more complex to just do # a simple text rendering. If there is any reason why we should have # those column in the mapper, then we should use the clean way. append(SQLExpression(self, where_expression=' AND '.join( where_pattern % (x, ) for x in join_table_list ))) self.sql_expression_list = sql_expression_list return SQLExpression( self, table_alias_dict=column_map.getTableAliasDict(), from_expression=self.from_expression, order_by_list=self.order_by_list, group_by_list=self.group_by_list, select_dict=self.final_select_dict, limit=self.limit, where_expression_operator='and', sql_expression_list=self.sql_expression_list) verifyClass(IEntireQuery, EntireQuery)