summaryrefslogtreecommitdiff
path: root/public/utils.py
diff options
context:
space:
mode:
Diffstat (limited to 'public/utils.py')
-rw-r--r--public/utils.py99
1 files changed, 78 insertions, 21 deletions
diff --git a/public/utils.py b/public/utils.py
index 2801c939..11091883 100644
--- a/public/utils.py
+++ b/public/utils.py
@@ -1,32 +1,89 @@
+from collections import defaultdict
from operator import attrgetter
-from main.models import Arch, Package
-from main.utils import cache_function
+from main.models import Arch, Repo, Package
+from main.utils import groupby_preserve_order, PackageStandin
+
+class RecentUpdate(object):
+ def __init__(self, packages):
+ if len(packages) == 0:
+ raise Exception
+ first = packages[0]
+ self.pkgbase = first.pkgbase
+ self.repo = first.repo
+ self.version = ''
+ self.classes = set()
+
+ self.classes.add(self.repo.name.lower())
+ if self.repo.testing:
+ self.classes.add('testing')
+ if self.repo.staging:
+ self.classes.add('staging')
+
+ packages = sorted(packages, key=attrgetter('arch', 'pkgname'))
+ # split the packages into two lists. we need to prefer packages
+ # matching pkgbase as our primary, and group everything else in other.
+ self.packages = [pkg for pkg in packages if pkg.pkgname == pkg.pkgbase]
+ self.others = [pkg for pkg in packages if pkg.pkgname != pkg.pkgbase]
+
+ if self.packages:
+ version = self.packages[0].full_version
+ if all(version == pkg.full_version for pkg in self.packages):
+ self.version = version
+ elif self.others:
+ version = self.others[0].full_version
+ if all(version == pkg.full_version for pkg in self.others):
+ self.version = version
+
+ def package_links(self):
+ '''Returns either actual packages or package-standins for virtual
+ pkgbase packages.'''
+ if self.packages:
+ # we have real packages- just yield each in sequence
+ for package in self.packages:
+ yield package
+ else:
+ # fake out the template- this is slightly hacky but yields one
+ # 'package-like' object per arch which is what the normal loop does
+ by_arch = defaultdict(list)
+ for package in self.others:
+ by_arch[package.arch].append(package)
+ for arch, packages in by_arch.items():
+ if len(packages) == 1:
+ yield packages[0]
+ else:
+ yield PackageStandin(packages[0])
+
+ def __unicode__(self):
+ return "RecentUpdate '%s %s' <%d packages>" % (
+ self.pkgbase, self.version, len(self.packages))
+
+def get_recent_updates(number=15, testing=True, staging=False):
+ repos = Repo.objects.all()
+ if not testing:
+ repos = repos.exclude(testing=True)
+ if not staging:
+ repos = repos.exclude(staging=True)
-@cache_function(300)
-def get_recent_updates():
# This is a bit of magic. We are going to show 15 on the front page, but we
# want to try and eliminate cross-architecture wasted space. Pull enough
# packages that we can later do some screening and trim out the fat.
pkgs = []
+ # grab a few extra so we can hopefully catch everything we need
+ fetch = number * 6
for arch in Arch.objects.all():
- # grab a few extra so we can hopefully catch everything we need
- pkgs += list(Package.objects.select_related(
- 'arch', 'repo').filter(arch=arch).order_by('-last_update')[:50])
- pkgs.sort(key=lambda q: q.last_update)
+ pkgs += list(Package.objects.normal().filter(
+ arch=arch, repo__in=repos).order_by('-last_update')[:fetch])
+ pkgs.sort(key=attrgetter('last_update'), reverse=True)
+
+ same_pkgbase_key = lambda x: (x.repo.name, x.pkgbase)
+ grouped = groupby_preserve_order(pkgs, same_pkgbase_key)
+
updates = []
- ctr = 0
- while ctr < 15 and len(pkgs) > 0:
- # not particularly happy with this logic, but it works.
- p = pkgs.pop()
- is_same = lambda q: p.is_same_version(q) and p.repo == q.repo
- samepkgs = filter(is_same, pkgs)
- samepkgs.append(p)
- samepkgs.sort(key=attrgetter('arch.name'))
- updates.append(samepkgs)
- for q in samepkgs:
- if p != q: pkgs.remove(q)
- ctr += 1
- return updates
+ for group in grouped:
+ update = RecentUpdate(group)
+ updates.append(update)
+
+ return updates[:number]
# vim: set ts=4 sw=4 et: