Реализация реализации агрегатов в SQL в Django 1.8

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

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

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')

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

1 ответ

Решение

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

    sortx = BinSortSQL(col, offset=x, bincount=xbin, max=x1)
    sorty = BinSortSQL(col, offset=y, bincount=ybin, max=y1)
Другие вопросы по тегам