diff options
Diffstat (limited to 'public/utils.py')
-rw-r--r-- | public/utils.py | 99 |
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: |