summary.py
141 lines
| 6.1 KiB
| text/x-python
|
PythonLexer
r252 | #!/usr/bin/env python | |||
# encoding: utf-8 | ||||
# summary controller for pylons | ||||
# Copyright (C) 2009-2010 Marcin Kuzminski <marcin@python-works.com> | ||||
r362 | # | |||
r252 | # This program is free software; you can redistribute it and/or | |||
# modify it under the terms of the GNU General Public License | ||||
# as published by the Free Software Foundation; version 2 | ||||
# of the License or (at your opinion) any later version of the license. | ||||
# | ||||
# This program is distributed in the hope that it will be useful, | ||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||
# GNU General Public License for more details. | ||||
# | ||||
# You should have received a copy of the GNU General Public License | ||||
# along with this program; if not, write to the Free Software | ||||
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, | ||||
# MA 02110-1301, USA. | ||||
""" | ||||
Created on April 18, 2010 | ||||
summary controller for pylons | ||||
@author: marcink | ||||
""" | ||||
r362 | from datetime import datetime, timedelta | |||
r232 | from pylons import tmpl_context as c, request | |||
r318 | from pylons_app.lib.auth import LoginRequired, HasRepoPermissionAnyDecorator | |||
r245 | from pylons_app.lib.base import BaseController, render | |||
r362 | from pylons_app.lib.helpers import person | |||
from pylons_app.lib.utils import OrderedDict | ||||
r282 | from pylons_app.model.hg_model import HgModel | |||
r362 | from time import mktime | |||
r282 | from webhelpers.paginate import Page | |||
r362 | import calendar | |||
r191 | import logging | |||
r362 | ||||
Marcin Kuzminski
|
r82 | log = logging.getLogger(__name__) | ||
class SummaryController(BaseController): | ||||
r191 | ||||
@LoginRequired() | ||||
r318 | @HasRepoPermissionAnyDecorator('repository.read', 'repository.write', | |||
'repository.admin') | ||||
Marcin Kuzminski
|
r82 | def __before__(self): | ||
r191 | super(SummaryController, self).__before__() | |||
r318 | ||||
Marcin Kuzminski
|
r82 | def index(self): | ||
hg_model = HgModel() | ||||
c.repo_info = hg_model.get_repo(c.repo_name) | ||||
r282 | c.repo_changesets = Page(list(c.repo_info[:10]), page=1, items_per_page=20) | |||
Marcin Kuzminski
|
r82 | e = request.environ | ||
r191 | uri = u'%(protocol)s://%(user)s@%(host)s/%(repo_name)s' % { | |||
r232 | 'protocol': e.get('wsgi.url_scheme'), | |||
'user':str(c.hg_app_user.username), | ||||
'host':e.get('HTTP_HOST'), | ||||
'repo_name':c.repo_name, } | ||||
r191 | c.clone_repo_url = uri | |||
r389 | c.repo_tags = OrderedDict() | |||
r277 | for name, hash in c.repo_info.tags.items()[:10]: | |||
c.repo_tags[name] = c.repo_info.get_changeset(hash) | ||||
r389 | c.repo_branches = OrderedDict() | |||
r277 | for name, hash in c.repo_info.branches.items()[:10]: | |||
c.repo_branches[name] = c.repo_info.get_changeset(hash) | ||||
r362 | ||||
c.commit_data = self.__get_commit_stats(c.repo_info) | ||||
r191 | return render('summary/summary.html') | |||
r362 | ||||
def __get_commit_stats(self, repo): | ||||
aggregate = OrderedDict() | ||||
#graph range | ||||
r390 | td = datetime.today() + timedelta(days=1) | |||
r362 | y = td.year | |||
m = td.month | ||||
d = td.day | ||||
r410 | c.ts_min = mktime((y, (td - timedelta(days=calendar.mdays[m] - 1)).month, | |||
d, 0, 0, 0, 0, 0, 0,)) | ||||
r362 | c.ts_max = mktime((y, m, d, 0, 0, 0, 0, 0, 0,)) | |||
r365 | ||||
r362 | ||||
def author_key_cleaner(k): | ||||
k = person(k) | ||||
return k | ||||
for cs in repo: | ||||
k = '%s-%s-%s' % (cs.date.timetuple()[0], cs.date.timetuple()[1], | ||||
cs.date.timetuple()[2]) | ||||
timetupple = [int(x) for x in k.split('-')] | ||||
timetupple.extend([0 for _ in xrange(6)]) | ||||
k = mktime(timetupple) | ||||
if aggregate.has_key(author_key_cleaner(cs.author)): | ||||
if aggregate[author_key_cleaner(cs.author)].has_key(k): | ||||
r410 | aggregate[author_key_cleaner(cs.author)][k]["commits"] += 1 | |||
aggregate[author_key_cleaner(cs.author)][k]["added"] += len(cs.added) | ||||
aggregate[author_key_cleaner(cs.author)][k]["changed"] += len(cs.changed) | ||||
aggregate[author_key_cleaner(cs.author)][k]["removed"] += len(cs.removed) | ||||
r362 | else: | |||
#aggregate[author_key_cleaner(cs.author)].update(dates_range) | ||||
if k >= c.ts_min and k <= c.ts_max: | ||||
r410 | aggregate[author_key_cleaner(cs.author)][k] = {} | |||
aggregate[author_key_cleaner(cs.author)][k]["commits"] = 1 | ||||
aggregate[author_key_cleaner(cs.author)][k]["added"] = len(cs.added) | ||||
aggregate[author_key_cleaner(cs.author)][k]["changed"] = len(cs.changed) | ||||
aggregate[author_key_cleaner(cs.author)][k]["removed"] = len(cs.removed) | ||||
r362 | else: | |||
if k >= c.ts_min and k <= c.ts_max: | ||||
aggregate[author_key_cleaner(cs.author)] = OrderedDict() | ||||
#aggregate[author_key_cleaner(cs.author)].update(dates_range) | ||||
r410 | aggregate[author_key_cleaner(cs.author)][k] = {} | |||
aggregate[author_key_cleaner(cs.author)][k]["commits"] = 1 | ||||
aggregate[author_key_cleaner(cs.author)][k]["added"] = len(cs.added) | ||||
aggregate[author_key_cleaner(cs.author)][k]["changed"] = len(cs.changed) | ||||
aggregate[author_key_cleaner(cs.author)][k]["removed"] = len(cs.removed) | ||||
r362 | ||||
d = '' | ||||
tmpl0 = u""""%s":%s""" | ||||
r410 | tmpl1 = u"""{label:"%s",data:%s,schema:["commits"]},""" | |||
r362 | for author in aggregate: | |||
r410 | ||||
r362 | d += tmpl0 % (author.decode('utf8'), | |||
tmpl1 \ | ||||
% (author.decode('utf8'), | ||||
r410 | [{"time":x, | |||
"commits":aggregate[author][x]['commits'], | ||||
"added":aggregate[author][x]['added'], | ||||
"changed":aggregate[author][x]['changed'], | ||||
"removed":aggregate[author][x]['removed'], | ||||
} for x in aggregate[author]])) | ||||
r362 | if d == '': | |||
r406 | d = '"%s":{label:"%s",data:[[0,1],]}' \ | |||
r362 | % (author_key_cleaner(repo.contact), | |||
author_key_cleaner(repo.contact)) | ||||
return d | ||||