Below I've pasted my first attempt at an aggregate function class. Its __init__ takes a queryset (and an optional list of aggregate functions to perform if, say, you only want the "sum"s rather than min/max/avg/sum). Thus you should be able to do things like
>>>class Foo(Model): ... blah = CharField(maxlength=42) ... xxx = IntegerField() ... yyy = SmallIntegerField() ... zzz = PositiveIntegerField() ... [add some data to Foo] >>> stats = Aggregate(Foo.objects.filter(xxx__gt=21)) The "stats" object now has properties xxx_sum xxx_max xxx_min xxx_average yyy_sum yyy_max yyy_min yyy_average zzz_sum zzz_max zzz_min zzz_average which can be accessed to provide the associated stats for the given model. If you make your own field types, if they're numeric, you'll want to add them to the set(). The constructor can also be called with the list of functions you want: >>> stats = Aggregate(Foo.objects.all(), ['sum']) in case you just want the *_sum properties. Future enhancements might include aggregates that don't involve sum/avg, so one might get the min/max of a date/time/char field. I haven't yet figured out a way to suppress the order_by portion, so what's currently in there is an ugly hack. But it would need to prevent the standard methods from inserting an ORDER BY clause against a non-aggregated field. At the moment, it meets my needs, but I'd gladly consider any feedback on this first draft version of things. Also, if you find it useful, feel free to shred it into your own purposes as desired. -tkc ############################################# from django.db import backend, connection NUMERIC_FIELDS = set(( 'FloatField', 'IntegerField', 'PositiveIntegerField', 'PositiveSmallIntegerField', 'SmallIntegerField', )) FUNCTIONS = [ 'min', 'max', 'average', 'sum', ] def is_numeric_field_type(field): return field.get_internal_type() in NUMERIC_FIELDS class Aggregate(object): def __init__(self, queryset, only_type=None): self._cache = None self.queryset = queryset.select_related(False) self.only_type = only_type or FUNCTIONS def all(self): self._get_results() def _get_aggregate_sql(self): q = self.queryset._clone() q._order_by = '?' # can't use None as it gets # overridden if a default Meta.ordering is specified # Should do some magic to prevent it from adding # the Meta.ordering if possible select, sql, params = q._get_sql_clause() meta = self.queryset.model._meta # build the SELECT contents: # fn(table.field) AS field_fn selects = [[ "%s(%s.%s) AS %s" % ( fn.lower(), backend.quote_name(meta.db_table), backend.quote_name(f.column), backend.quote_name("%s_%s" % (f.column, fn.lower())), ) for f in meta.fields if is_numeric_field_type(f) ] for fn in self.only_type] # the sum(selects, []) flattens # the list-of-lists into just a list return sum(selects, []), sql, params def _get_results(self): if self._cache is None: select, sql, params = self._get_aggregate_sql() cursor = connection.cursor() sql = "SELECT " + ",".join(select) + sql cursor.execute(sql, params) results = cursor.fetchone() field_names = [d[0] for d in cursor.description] self._cache = dict(zip(field_names, results)) return self._cache def __getattribute__(self, k): try: return object.__getattribute__(self, k) except: results = self._get_results() return results[k] --~--~---------~--~----~------------~-------~--~----~ You received this message because you are subscribed to the Google Groups "Django users" group. To post to this group, send email to django-users@googlegroups.com To unsubscribe from this group, send email to [EMAIL PROTECTED] For more options, visit this group at http://groups.google.com/group/django-users?hl=en -~----------~----~----~----~------~----~------~--~---