2015-08-14 2 views
1

Я работаю над обновлением существующей базы кода, использующей Django 1.6 для Django 1.8. В процессе я столкнулся с конкретной проблемой с агрегатами. В этом коде PGDAggregate класс имеет метод add_to_query, который предназначен для конкретизации реализации SQL агрегата и устанавливает его в качестве переменной класса (агрегат), который я бы использовать, чтобы вызвать as_sql метод по умолчанию SQL Aggregate (django/db.models.sql.aggregates.Aggregate) из другого файла.Выполнение SQL-реализации агрегатов в Django 1.8

Мой код (Первый файл, как я реализую агрегат):

from django.db.models.aggregates import Aggregate 
from django.db.models.sql.aggregates import Aggregate as SQLAggregate 

class PGDAggregate(Aggregate): 
    """ 
    Modified to allow Aggregate functions outside of the Django module 
    """ 

    def add_to_query(self, query, alias, col, source, is_summary): 
     """Add the aggregate to the nominated query. 

     This method is used to convert the generic Aggregate definition into a 
     backend-specific definition. 

     * query is the backend-specific query instance to which the aggregate 
      is to be added. 
     * col is a column reference describing the subject field 
      of the aggregate. It can be an alias, or a tuple describing 
      a table and column name. 
     * source is the underlying field or aggregate definition for 
      the column reference. If the aggregate is not an ordinal or 
      computed type, this reference is used to determine the coerced 
      output type of the aggregate. 
     * is_summary is a boolean that is set True if the aggregate is a 
      summary value rather than an annotation. 
     """ 
     klass = globals()['%sSQL' % self.name] 
     aggregate = klass(col, source=source, is_summary=is_summary, **self.extra) 

     # Validate that the backend has a fully supported, correct 
     # implementation of this aggregate 
     query.aggregates[alias] = aggregate 
     self.aggregate = aggregate 


class BinSort(PGDAggregate): 
    alias = 'BinSort' 
    name = 'BinSort' 


class BinSortSQL(SQLAggregate): 
    sql_function = '' 
    sql_template = '%(function)sFLOOR((IF(%(field)s<%(offset).16f,360,0)+%(field)s-%(offset).16f)/%(bincount).16f)-IF(%(field)s=%(max).16f,1,0)' 

Это, как я пытаюсь использовать совокупный атрибут (экземпляр по умолчанию SQL Совокупности) из второго файла для вызова as_sql метод.

sortx = BinSort(xTextString, offset=x, bincount=xbin, max=x1) 
sorty = BinSort(yTextString, offset=y, bincount=ybin, max=y1) 
annotated_query.annotate(x=sortx, y=sorty) 
cn = connections['default'] 
qn = SQLCompiler(annotated_query.query, cn, 'default').quote_name_unless_alias 
sortx_sql = sortx.aggregate.as_sql(qn, cn)[0] 
sorty_sql = sorty.aggregate.as_sql(qn, cn)[0] 

ошибка, что я получаю (в литрах: 6) в этом реализация,

исключение объекта 'BinSort' не имеет атрибут 'заполнитель'

Как шаги отладки, я попытался проверить, имеет ли экземпляр BinSort атрибут «агрегат», используя

hasattr(sortx, 'aggregate') 

, который вернул мне False. Но когда я пытаюсь проверить, распечатывая атрибут aggregate из метода add_to_query, я мог бы очень сильно увидеть, что атрибут печатается. Кроме того, я реализовал это так, как указано в Django 1.8 doc, https://github.com/django/django/blob/stable/1.8.x/django/db/models/aggregates.py#L46

ответ

0

Хотя это не является решением для объяснения неожиданного поведения, но это работает. Поскольку я хотел использовать метод as_sql() класса SQL Aggregate по умолчанию, я инициализировал BinSortSQL непосредственно вместо класса BinSort и использовал его метод as_sql().

sortx = BinSortSQL(col, offset=x, bincount=xbin, max=x1) 
    sorty = BinSortSQL(col, offset=y, bincount=ybin, max=y1) 
Смежные вопросы