Files
@ a08f610e545e
Branch filter:
Location: kallithea/pylons_app/lib/celerylib/tasks.py - annotation
a08f610e545e
11.2 KiB
text/x-python
Implemented server side forks
added ability to pass session to user/repo models
added ability to pass session to user/repo models
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 | 3fc3ce53659b a3d9d24acbec a9e50dce3081 d5efb83590ef a3d9d24acbec ac32a026c306 3fc3ce53659b a3d9d24acbec 3fc3ce53659b d5efb83590ef b12ea84fb906 d5efb83590ef a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec 3fc3ce53659b a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec 3fc3ce53659b a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec d280aa1c85c6 a3d9d24acbec a3d9d24acbec d280aa1c85c6 d280aa1c85c6 fb0c3af6031b a3d9d24acbec ac32a026c306 2256c78afe53 2256c78afe53 a3d9d24acbec d5efb83590ef d5efb83590ef 2256c78afe53 2256c78afe53 a9e50dce3081 2256c78afe53 2256c78afe53 2256c78afe53 b50e79b4257a 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 5c376ac2d4c9 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 b50e79b4257a b50e79b4257a 3fc3ce53659b d5efb83590ef b50e79b4257a b50e79b4257a 2256c78afe53 b12ea84fb906 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 d5efb83590ef 2256c78afe53 2256c78afe53 3fc3ce53659b 3fc3ce53659b 3fc3ce53659b 3fc3ce53659b 3fc3ce53659b 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 3fc3ce53659b 3fc3ce53659b 2256c78afe53 b12ea84fb906 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 3fc3ce53659b 3fc3ce53659b 2256c78afe53 b12ea84fb906 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 3fc3ce53659b 2256c78afe53 2256c78afe53 2256c78afe53 5c376ac2d4c9 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 5c376ac2d4c9 2256c78afe53 5c376ac2d4c9 b12ea84fb906 b12ea84fb906 2256c78afe53 2256c78afe53 5c376ac2d4c9 5c376ac2d4c9 5c376ac2d4c9 5c376ac2d4c9 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 d5efb83590ef d5efb83590ef fb0c3af6031b 2256c78afe53 a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 2256c78afe53 a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a9e50dce3081 a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec a3d9d24acbec d5efb83590ef a08f610e545e a08f610e545e a08f610e545e a08f610e545e a08f610e545e a08f610e545e a08f610e545e a08f610e545e a08f610e545e a08f610e545e a08f610e545e a08f610e545e a08f610e545e a08f610e545e a08f610e545e a08f610e545e a08f610e545e a08f610e545e d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef d5efb83590ef | from celery.decorators import task
from celery.task.sets import subtask
from celeryconfig import PYLONS_CONFIG as config
from operator import itemgetter
from pylons.i18n.translation import _
from pylons_app.lib.celerylib import run_task, locked_task
from pylons_app.lib.helpers import person
from pylons_app.lib.smtp_mailer import SmtpMailer
from pylons_app.lib.utils import OrderedDict
from time import mktime
from vcs.backends.hg import MercurialRepository
import json
import traceback
__all__ = ['whoosh_index', 'get_commits_stats',
'reset_user_password', 'send_email']
def get_session():
from sqlalchemy import engine_from_config
from sqlalchemy.orm import sessionmaker, scoped_session
engine = engine_from_config(dict(config.items('app:main')), 'sqlalchemy.db1.')
sa = scoped_session(sessionmaker(bind=engine))
return sa
def get_hg_settings():
from pylons_app.model.db import HgAppSettings
try:
sa = get_session()
ret = sa.query(HgAppSettings).all()
finally:
sa.remove()
if not ret:
raise Exception('Could not get application settings !')
settings = {}
for each in ret:
settings['hg_app_' + each.app_settings_name] = each.app_settings_value
return settings
def get_hg_ui_settings():
from pylons_app.model.db import HgAppUi
try:
sa = get_session()
ret = sa.query(HgAppUi).all()
finally:
sa.remove()
if not ret:
raise Exception('Could not get application ui settings !')
settings = {}
for each in ret:
k = each.ui_key
v = each.ui_value
if k == '/':
k = 'root_path'
if k.find('.') != -1:
k = k.replace('.', '_')
if each.ui_section == 'hooks':
v = each.ui_active
settings[each.ui_section + '_' + k] = v
return settings
@task
@locked_task
def whoosh_index(repo_location, full_index):
log = whoosh_index.get_logger()
from pylons_app.lib.indexers.daemon import WhooshIndexingDaemon
WhooshIndexingDaemon(repo_location=repo_location).run(full_index=full_index)
@task
@locked_task
def get_commits_stats(repo_name, ts_min_y, ts_max_y):
from pylons_app.model.db import Statistics, Repository
log = get_commits_stats.get_logger()
author_key_cleaner = lambda k: person(k).replace('"', "") #for js data compatibilty
commits_by_day_author_aggregate = {}
commits_by_day_aggregate = {}
repos_path = get_hg_ui_settings()['paths_root_path'].replace('*', '')
repo = MercurialRepository(repos_path + repo_name)
skip_date_limit = True
parse_limit = 350 #limit for single task changeset parsing optimal for
last_rev = 0
last_cs = None
timegetter = itemgetter('time')
sa = get_session()
dbrepo = sa.query(Repository)\
.filter(Repository.repo_name == repo_name).scalar()
cur_stats = sa.query(Statistics)\
.filter(Statistics.repository == dbrepo).scalar()
if cur_stats:
last_rev = cur_stats.stat_on_revision
if not repo.revisions:
return True
if last_rev == repo.revisions[-1] and len(repo.revisions) > 1:
#pass silently without any work if we're not on first revision or current
#state of parsing revision(from db marker) is the last revision
return True
if cur_stats:
commits_by_day_aggregate = OrderedDict(
json.loads(
cur_stats.commit_activity_combined))
commits_by_day_author_aggregate = json.loads(cur_stats.commit_activity)
log.debug('starting parsing %s', parse_limit)
for cnt, rev in enumerate(repo.revisions[last_rev:]):
last_cs = cs = repo.get_changeset(rev)
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 commits_by_day_author_aggregate.has_key(author_key_cleaner(cs.author)):
try:
l = [timegetter(x) for x in commits_by_day_author_aggregate\
[author_key_cleaner(cs.author)]['data']]
time_pos = l.index(k)
except ValueError:
time_pos = False
if time_pos >= 0 and time_pos is not False:
datadict = commits_by_day_author_aggregate\
[author_key_cleaner(cs.author)]['data'][time_pos]
datadict["commits"] += 1
datadict["added"] += len(cs.added)
datadict["changed"] += len(cs.changed)
datadict["removed"] += len(cs.removed)
#print datadict
else:
#print 'ELSE !!!!'
if k >= ts_min_y and k <= ts_max_y or skip_date_limit:
datadict = {"time":k,
"commits":1,
"added":len(cs.added),
"changed":len(cs.changed),
"removed":len(cs.removed),
}
commits_by_day_author_aggregate\
[author_key_cleaner(cs.author)]['data'].append(datadict)
else:
#print k, 'nokey ADDING'
if k >= ts_min_y and k <= ts_max_y or skip_date_limit:
commits_by_day_author_aggregate[author_key_cleaner(cs.author)] = {
"label":author_key_cleaner(cs.author),
"data":[{"time":k,
"commits":1,
"added":len(cs.added),
"changed":len(cs.changed),
"removed":len(cs.removed),
}],
"schema":["commits"],
}
# #gather all data by day
if commits_by_day_aggregate.has_key(k):
commits_by_day_aggregate[k] += 1
else:
commits_by_day_aggregate[k] = 1
if cnt >= parse_limit:
#don't fetch to much data since we can freeze application
break
overview_data = []
for k, v in commits_by_day_aggregate.items():
overview_data.append([k, v])
overview_data = sorted(overview_data, key=itemgetter(0))
if not commits_by_day_author_aggregate:
commits_by_day_author_aggregate[author_key_cleaner(repo.contact)] = {
"label":author_key_cleaner(repo.contact),
"data":[0, 1],
"schema":["commits"],
}
stats = cur_stats if cur_stats else Statistics()
stats.commit_activity = json.dumps(commits_by_day_author_aggregate)
stats.commit_activity_combined = json.dumps(overview_data)
log.debug('last revison %s', last_rev)
leftovers = len(repo.revisions[last_rev:])
log.debug('revisions to parse %s', leftovers)
if last_rev == 0 or leftovers < parse_limit:
stats.languages = json.dumps(__get_codes_stats(repo_name))
stats.repository = dbrepo
stats.stat_on_revision = last_cs.revision
try:
sa.add(stats)
sa.commit()
except:
log.error(traceback.format_exc())
sa.rollback()
return False
if len(repo.revisions) > 1:
run_task(get_commits_stats, repo_name, ts_min_y, ts_max_y)
return True
@task
def reset_user_password(user_email):
log = reset_user_password.get_logger()
from pylons_app.lib import auth
from pylons_app.model.db import User
try:
try:
sa = get_session()
user = sa.query(User).filter(User.email == user_email).scalar()
new_passwd = auth.PasswordGenerator().gen_password(8,
auth.PasswordGenerator.ALPHABETS_BIG_SMALL)
if user:
user.password = auth.get_crypt_password(new_passwd)
sa.add(user)
sa.commit()
log.info('change password for %s', user_email)
if new_passwd is None:
raise Exception('unable to generate new password')
except:
log.error(traceback.format_exc())
sa.rollback()
run_task(send_email, user_email,
"Your new hg-app password",
'Your new hg-app password:%s' % (new_passwd))
log.info('send new password mail to %s', user_email)
except:
log.error('Failed to update user password')
log.error(traceback.format_exc())
return True
@task
def send_email(recipients, subject, body):
log = send_email.get_logger()
email_config = dict(config.items('DEFAULT'))
mail_from = email_config.get('app_email_from')
user = email_config.get('smtp_username')
passwd = email_config.get('smtp_password')
mail_server = email_config.get('smtp_server')
mail_port = email_config.get('smtp_port')
tls = email_config.get('smtp_use_tls')
ssl = False
try:
m = SmtpMailer(mail_from, user, passwd, mail_server,
mail_port, ssl, tls)
m.send(recipients, subject, body)
except:
log.error('Mail sending failed')
log.error(traceback.format_exc())
return False
return True
@task
def create_repo_fork(form_data, cur_user):
import os
from pylons_app.lib.utils import invalidate_cache
from pylons_app.model.repo_model import RepoModel
sa = get_session()
rm = RepoModel(sa)
rm.create(form_data, cur_user, just_db=True, fork=True)
repos_path = get_hg_ui_settings()['paths_root_path'].replace('*', '')
repo_path = os.path.join(repos_path, form_data['repo_name'])
repo_fork_path = os.path.join(repos_path, form_data['fork_name'])
MercurialRepository(str(repo_fork_path), True, clone_url=str(repo_path))
#invalidate_cache('cached_repo_list')
def __get_codes_stats(repo_name):
LANGUAGES_EXTENSIONS = ['action', 'adp', 'ashx', 'asmx', 'aspx', 'asx', 'axd', 'c',
'cfg', 'cfm', 'cpp', 'cs', 'diff', 'do', 'el', 'erl',
'h', 'java', 'js', 'jsp', 'jspx', 'lisp',
'lua', 'm', 'mako', 'ml', 'pas', 'patch', 'php', 'php3',
'php4', 'phtml', 'pm', 'py', 'rb', 'rst', 's', 'sh',
'tpl', 'txt', 'vim', 'wss', 'xhtml', 'xml', 'xsl', 'xslt',
'yaws']
repos_path = get_hg_ui_settings()['paths_root_path'].replace('*', '')
repo = MercurialRepository(repos_path + repo_name)
code_stats = {}
for topnode, dirs, files in repo.walk('/', 'tip'):
for f in files:
k = f.mimetype
if f.extension in LANGUAGES_EXTENSIONS:
if code_stats.has_key(k):
code_stats[k] += 1
else:
code_stats[k] = 1
return code_stats or {}
|