##// END OF EJS Templates
#47 added editing of groups, and moving them between. Added check constraint for groups...
#47 added editing of groups, and moving them between. Added check constraint for groups to prevent of loops

File last commit:

r1343:a04fe598 beta
r1349:526120c7 beta
Show More
scm.py
442 lines | 15.3 KiB | text/x-python | PythonLexer
rolled back to make transient since got some exceptions on expunge...
r757 # -*- coding: utf-8 -*-
"""
docs updates
r811 rhodecode.model.scm
~~~~~~~~~~~~~~~~~~~
rolled back to make transient since got some exceptions on expunge...
r757
docs updates
r811 Scm model for RhodeCode
rolled back to make transient since got some exceptions on expunge...
r757 :created_on: Apr 9, 2010
:author: marcink
source code cleanup: remove trailing white space, normalize file endings
r1203 :copyright: (C) 2009-2011 Marcin Kuzminski <marcin@python-works.com>
rolled back to make transient since got some exceptions on expunge...
r757 :license: GPLv3, see COPYING for more details.
"""
fixed license issue #149
r1206 # 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, either version 3 of the License, or
# (at your option) any later version.
source code cleanup: remove trailing white space, normalize file endings
r1203 #
Refactor codes for scm model...
r691 # 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.
source code cleanup: remove trailing white space, normalize file endings
r1203 #
Refactor codes for scm model...
r691 # You should have received a copy of the GNU General Public License
fixed license issue #149
r1206 # along with this program. If not, see <http://www.gnu.org/licenses/>.
rolled back to make transient since got some exceptions on expunge...
r757 import os
import time
import traceback
import logging
Code refactor number 2
r1022 from mercurial import ui
from sqlalchemy.exc import DatabaseError
another major codes rewrite:...
r1045 from sqlalchemy.orm import make_transient
Code refactor number 2
r1022
from beaker.cache import cache_region, region_invalidate
rolled back to make transient since got some exceptions on expunge...
r757 from vcs import get_backend
from vcs.utils.helpers import get_scm
from vcs.exceptions import RepositoryError, VCSError
from vcs.utils.lazy import LazyProperty
moved out commit into scm model, and added cache invalidation after commit.
r1311 from vcs.nodes import FileNode
rolled back to make transient since got some exceptions on expunge...
r757
Disable git support due to large problems with dulwich....
r710 from rhodecode import BACKENDS
Refactor codes for scm model...
r691 from rhodecode.lib import helpers as h
from rhodecode.lib.auth import HasRepoPermissionAny
another major code rafactor, reimplemented (almost from scratch)...
r1038 from rhodecode.lib.utils import get_repos as get_filesystem_repos, make_ui, \
action_logger
fixed Example celery config to ampq,...
r752 from rhodecode.model import BaseModel
code cleanups
r758 from rhodecode.model.user import UserModel
another major code rafactor, reimplemented (almost from scratch)...
r1038 from rhodecode.model.repo import RepoModel
code cleanups
r758 from rhodecode.model.db import Repository, RhodeCodeUi, CacheInvalidation, \
Implemented fancier top menu for logged and anonymous users...
r784 UserFollowing, UserLog
#50 on point cache invalidation changes....
r692 from rhodecode.model.caching_query import FromCache
rolled back to make transient since got some exceptions on expunge...
r757
Refactor codes for scm model...
r691 log = logging.getLogger(__name__)
rolled back to make transient since got some exceptions on expunge...
r757
added action loggers to following repositories,...
r735 class UserTemp(object):
def __init__(self, user_id):
self.user_id = user_id
Fixed repo of Temp user and repo for better logging
r901
def __repr__(self):
return "<%s('id:%s')>" % (self.__class__.__name__, self.user_id)
#150 fixes for errors on repositories mapped in db but corrupted in filesystem
r1213
added action loggers to following repositories,...
r735 class RepoTemp(object):
def __init__(self, repo_id):
self.repo_id = repo_id
Added icons with numbers of followers and number of forks
r747
Fixed repo of Temp user and repo for better logging
r901 def __repr__(self):
return "<%s('id:%s')>" % (self.__class__.__name__, self.repo_id)
#150 fixes for errors on repositories mapped in db but corrupted in filesystem
r1213
fixed Example celery config to ampq,...
r752 class ScmModel(BaseModel):
docs updates
r811 """Generic Scm Model
Refactor codes for scm model...
r691 """
@LazyProperty
def repos_path(self):
docs updates
r811 """Get's the repositories root path from database
Refactor codes for scm model...
r691 """
docs updates
r811
Refactor codes for scm model...
r691 q = self.sa.query(RhodeCodeUi).filter(RhodeCodeUi.ui_key == '/').one()
return q.ui_value
another major code rafactor, reimplemented (almost from scratch)...
r1038 def repo_scan(self, repos_path=None):
source code cleanup: remove trailing white space, normalize file endings
r1203 """Listing of repositories in given path. This path should not be a
Refactor codes for scm model...
r691 repository itself. Return a dictionary of repository objects
source code cleanup: remove trailing white space, normalize file endings
r1203
Refactor codes for scm model...
r691 :param repos_path: path to directory containing repositories
"""
docs updates
r811
Refactor codes for scm model...
r691 log.info('scanning for repositories in %s', repos_path)
another major code rafactor, reimplemented (almost from scratch)...
r1038 if repos_path is None:
repos_path = self.repos_path
baseui = make_ui('db')
Refactor codes for scm model...
r691 repos_list = {}
Added recursive scanning for repositories in directory
r877 for name, path in get_filesystem_repos(repos_path, recursive=True):
Refactor codes for scm model...
r691 try:
#150 fixes for errors on repositories mapped in db but corrupted in filesystem
r1213 if name in repos_list:
Refactor codes for scm model...
r691 raise RepositoryError('Duplicate repository name %s '
'found in %s' % (name, path))
else:
klass = get_backend(path[0])
Disable git support due to large problems with dulwich....
r710 if path[0] == 'hg' and path[0] in BACKENDS.keys():
Refactor codes for scm model...
r691 repos_list[name] = klass(path[1], baseui=baseui)
Disable git support due to large problems with dulwich....
r710 if path[0] == 'git' and path[0] in BACKENDS.keys():
Refactor codes for scm model...
r691 repos_list[name] = klass(path[1])
except OSError:
continue
return repos_list
def get_repos(self, all_repos=None):
#47 implemented basic gui for browsing repo groups
r1343 """
Get all repos from db and for each repo create it's
#150 fixes for errors on repositories mapped in db but corrupted in filesystem
r1213 backend instance and fill that backed with information from database
source code cleanup: remove trailing white space, normalize file endings
r1203
#47 implemented basic gui for browsing repo groups
r1343 :param all_repos: list of repository names as strings
give specific repositories list, good for filtering
Refactor codes for scm model...
r691 """
bugfix, when user had no repos he would see all repos in my account, (correct commit)
r767 if all_repos is None:
Added repo group page showing what reposiories are inside a group
r1193 repos = self.sa.query(Repository)\
#47 implemented basic gui for browsing repo groups
r1343 .filter(Repository.group_id == None)\
Added repo group page showing what reposiories are inside a group
r1193 .order_by(Repository.repo_name).all()
all_repos = [r.repo_name for r in repos]
Refactor codes for scm model...
r691
fixed bug when invalidation was making to many queries when there was no list of invalidation
r791 #get the repositories that should be invalidated
cache list speed improvement
r726 invalidation_list = [str(x.cache_key) for x in \
self.sa.query(CacheInvalidation.cache_key)\
.filter(CacheInvalidation.cache_active == False)\
.all()]
another major codes rewrite:...
r1045 for r_name in all_repos:
r_dbr = self.get(r_name, invalidation_list)
bugfix for getting proper values from cache
r1042 if r_dbr is not None:
repo, dbrepo = r_dbr
another major codes rewrite:...
r1045
fixes issue #163 RhodeCode will check and report a corrupted repo. A rescan with destroy old data should be perform after such operations
r1255 if repo is None or dbrepo is None:
added some info to corrupted repo message that helps diagnose the cause from log itself.
r1265 log.error('Repository "%s" looks somehow corrupted '
'fs-repo:%s,db-repo:%s both values should be '
'present', r_name, repo, dbrepo)
#150 fixes for errors on repositories mapped in db but corrupted in filesystem
r1213 continue
Refactor codes for scm model...
r691 last_change = repo.last_change
tip = h.get_changeset_safe(repo, 'tip')
tmp_d = {}
another major codes rewrite:...
r1045 tmp_d['name'] = dbrepo.repo_name
Refactor codes for scm model...
r691 tmp_d['name_sort'] = tmp_d['name'].lower()
another major code rafactor, reimplemented (almost from scratch)...
r1038 tmp_d['description'] = dbrepo.description
Refactor codes for scm model...
r691 tmp_d['description_sort'] = tmp_d['description']
tmp_d['last_change'] = last_change
#150 fixes for errors on repositories mapped in db but corrupted in filesystem
r1213 tmp_d['last_change_sort'] = time.mktime(last_change \
.timetuple())
Refactor codes for scm model...
r691 tmp_d['tip'] = tip.raw_id
tmp_d['tip_sort'] = tip.revision
tmp_d['rev'] = tip.revision
another major code rafactor, reimplemented (almost from scratch)...
r1038 tmp_d['contact'] = dbrepo.user.full_contact
Refactor codes for scm model...
r691 tmp_d['contact_sort'] = tmp_d['contact']
fixed sorting in maing page by owners
r1005 tmp_d['owner_sort'] = tmp_d['contact']
Refactor codes for scm model...
r691 tmp_d['repo_archives'] = list(repo._get_archives())
tmp_d['last_msg'] = tip.message
tmp_d['repo'] = repo
another major codes rewrite:...
r1045 tmp_d['dbrepo'] = dbrepo.get_dict()
#150 fixes for errors on repositories mapped in db but corrupted in filesystem
r1213 tmp_d['dbrepo_fork'] = dbrepo.fork.get_dict() if dbrepo.fork \
else {}
Refactor codes for scm model...
r691 yield tmp_d
another major code rafactor, reimplemented (almost from scratch)...
r1038 def get(self, repo_name, invalidation_list=None, retval='all'):
"""Returns a tuple of Repository,DbRepository,
Get's repository from given name, creates BackendInstance and
Refactor codes for scm model...
r691 propagates it's data from database with all additional information
source code cleanup: remove trailing white space, normalize file endings
r1203
Refactor codes for scm model...
r691 :param repo_name:
fixed bug when invalidation was making to many queries when there was no list of invalidation
r791 :param invalidation_list: if a invalidation list is given the get
source code cleanup: remove trailing white space, normalize file endings
r1203 method should not manually check if this repository needs
fixed bug when invalidation was making to many queries when there was no list of invalidation
r791 invalidation and just invalidate the repositories in list
another major code rafactor, reimplemented (almost from scratch)...
r1038 :param retval: string specifing what to return one of 'repo','dbrepo',
'all'if repo or dbrepo is given it'll just lazy load chosen type
and return None as the second
Refactor codes for scm model...
r691 """
if not HasRepoPermissionAny('repository.read', 'repository.write',
'repository.admin')(repo_name, 'get repo check'):
return
fixed bug when invalidation was making to many queries when there was no list of invalidation
r791 #======================================================================
# CACHE FUNCTION
#======================================================================
#50 on point cache invalidation changes....
r692 @cache_region('long_term')
Refactor codes for scm model...
r691 def _get_repo(repo_name):
repo_path = os.path.join(self.repos_path, repo_name)
rolled back to make transient since got some exceptions on expunge...
r757
try:
alias = get_scm(repo_path)[0]
log.debug('Creating instance of %s repository', alias)
backend = get_backend(alias)
except VCSError:
log.error(traceback.format_exc())
another major code rafactor, reimplemented (almost from scratch)...
r1038 log.error('Perhaps this repository is in db and not in '
'filesystem run rescan repositories with '
'"destroy old data " option from admin panel')
rolled back to make transient since got some exceptions on expunge...
r757 return
Refactor codes for scm model...
r691
if alias == 'hg':
another major code rafactor, reimplemented (almost from scratch)...
r1038 repo = backend(repo_path, create=False, baseui=make_ui('db'))
Refactor codes for scm model...
r691 #skip hidden web repository
if repo._get_hidden():
return
else:
repo = backend(repo_path, create=False)
return repo
fixed bug introduced in latest commit
r792 pre_invalidate = True
another major code rafactor, reimplemented (almost from scratch)...
r1038 dbinvalidate = False
fixed bug introduced in latest commit
r792 if invalidation_list is not None:
pre_invalidate = repo_name in invalidation_list
if pre_invalidate:
another major code rafactor, reimplemented (almost from scratch)...
r1038 #this returns object to invalidate
fixed bug introduced in latest commit
r792 invalidate = self._should_invalidate(repo_name)
if invalidate:
log.info('invalidating cache for repository %s', repo_name)
extended admin rescan to show what repositories was added and what removed...
r1039 region_invalidate(_get_repo, None, repo_name)
fixed bug introduced in latest commit
r792 self._mark_invalidated(invalidate)
another major code rafactor, reimplemented (almost from scratch)...
r1038 dbinvalidate = True
fixed bug introduced in latest commit
r792
another major code rafactor, reimplemented (almost from scratch)...
r1038 r, dbr = None, None
if retval == 'repo' or 'all':
r = _get_repo(repo_name)
if retval == 'dbrepo' or 'all':
another major codes rewrite:...
r1045 dbr = RepoModel().get_full(repo_name, cache=True,
another major code rafactor, reimplemented (almost from scratch)...
r1038 invalidate=dbinvalidate)
return r, dbr
Refactor codes for scm model...
r691
#50 on point cache invalidation changes....
r692 def mark_for_invalidation(self, repo_name):
source code cleanup: remove trailing white space, normalize file endings
r1203 """Puts cache invalidation task into db for
#50 on point cache invalidation changes....
r692 further global cache invalidation
source code cleanup: remove trailing white space, normalize file endings
r1203
#50 on point cache invalidation changes....
r692 :param repo_name: this repo that should invalidation take place
"""
docs updates
r811
#50 on point cache invalidation changes....
r692 log.debug('marking %s for invalidation', repo_name)
cache = self.sa.query(CacheInvalidation)\
.filter(CacheInvalidation.cache_key == repo_name).scalar()
if cache:
#mark this cache as inactive
cache.cache_active = False
else:
log.debug('cache key not found in invalidation db -> creating one')
cache = CacheInvalidation(repo_name)
try:
self.sa.add(cache)
self.sa.commit()
code cleanups
r758 except (DatabaseError,):
#50 on point cache invalidation changes....
r692 log.error(traceback.format_exc())
self.sa.rollback()
implemented user dashboards, and following system.
r734 def toggle_following_repo(self, follow_repo_id, user_id):
#50 on point cache invalidation changes....
r692
implemented user dashboards, and following system.
r734 f = self.sa.query(UserFollowing)\
.filter(UserFollowing.follows_repo_id == follow_repo_id)\
.filter(UserFollowing.user_id == user_id).scalar()
if f is not None:
Added icons with numbers of followers and number of forks
r747
implemented user dashboards, and following system.
r734 try:
self.sa.delete(f)
self.sa.commit()
added action loggers to following repositories,...
r735 action_logger(UserTemp(user_id),
'stopped_following_repo',
Added icons with numbers of followers and number of forks
r747 RepoTemp(follow_repo_id))
implemented user dashboards, and following system.
r734 return
except:
log.error(traceback.format_exc())
self.sa.rollback()
raise
try:
f = UserFollowing()
f.user_id = user_id
f.follows_repo_id = follow_repo_id
self.sa.add(f)
self.sa.commit()
added action loggers to following repositories,...
r735 action_logger(UserTemp(user_id),
'started_following_repo',
Added icons with numbers of followers and number of forks
r747 RepoTemp(follow_repo_id))
implemented user dashboards, and following system.
r734 except:
log.error(traceback.format_exc())
self.sa.rollback()
raise
#150 fixes for errors on repositories mapped in db but corrupted in filesystem
r1213 def toggle_following_user(self, follow_user_id, user_id):
implemented user dashboards, and following system.
r734 f = self.sa.query(UserFollowing)\
.filter(UserFollowing.follows_user_id == follow_user_id)\
.filter(UserFollowing.user_id == user_id).scalar()
if f is not None:
try:
self.sa.delete(f)
self.sa.commit()
return
except:
log.error(traceback.format_exc())
self.sa.rollback()
raise
try:
f = UserFollowing()
f.user_id = user_id
f.follows_user_id = follow_user_id
self.sa.add(f)
self.sa.commit()
except:
log.error(traceback.format_exc())
self.sa.rollback()
raise
fixed following js snipet. It' can be called multiple times now next to each repository...
r999 def is_following_repo(self, repo_name, user_id, cache=False):
implemented user dashboards, and following system.
r734 r = self.sa.query(Repository)\
.filter(Repository.repo_name == repo_name).scalar()
f = self.sa.query(UserFollowing)\
.filter(UserFollowing.follows_repository == r)\
.filter(UserFollowing.user_id == user_id).scalar()
return f is not None
fixed following js snipet. It' can be called multiple times now next to each repository...
r999 def is_following_user(self, username, user_id, cache=False):
code cleanups
r758 u = UserModel(self.sa).get_by_username(username)
implemented user dashboards, and following system.
r734
f = self.sa.query(UserFollowing)\
.filter(UserFollowing.follows_user == u)\
.filter(UserFollowing.user_id == user_id).scalar()
return f is not None
#50 on point cache invalidation changes....
r692
Added icons with numbers of followers and number of forks
r747 def get_followers(self, repo_id):
fixed condition evaluated for gitrepo that returned null, simplified scm functions
r1282 if not isinstance(repo_id, int):
repo_id = getattr(Repository.by_repo_name(repo_id), 'repo_id')
return self.sa.query(UserFollowing)\
.filter(UserFollowing.follows_repo_id == repo_id).count()
Added icons with numbers of followers and number of forks
r747
def get_forks(self, repo_id):
fixed condition evaluated for gitrepo that returned null, simplified scm functions
r1282 if not isinstance(repo_id, int):
repo_id = getattr(Repository.by_repo_name(repo_id), 'repo_id')
return self.sa.query(Repository)\
Added icons with numbers of followers and number of forks
r747 .filter(Repository.fork_id == repo_id).count()
#50 on point cache invalidation changes....
r692
#109, added manual pull of changes for repositories that have remote location filled in....
r1114 def pull_changes(self, repo_name, username):
repo, dbrepo = self.get(repo_name, retval='all')
try:
#150 fixes for errors on repositories mapped in db but corrupted in filesystem
r1213 extras = {'ip': '',
'username': username,
'action': 'push_remote',
'repository': repo_name}
#109, added manual pull of changes for repositories that have remote location filled in....
r1114
source code cleanup: remove trailing white space, normalize file endings
r1203 #inject ui extra param to log this action via push logger
#109, added manual pull of changes for repositories that have remote location filled in....
r1114 for k, v in extras.items():
repo._repo.ui.setconfig('rhodecode_extras', k, v)
repo.pull(dbrepo.clone_uri)
self.mark_for_invalidation(repo_name)
except:
log.error(traceback.format_exc())
raise
moved out commit into scm model, and added cache invalidation after commit.
r1311
logged local commit with special action via action_logger,
r1312 def commit_change(self, repo, repo_name, cs, user, author, message, content,
moved out commit into scm model, and added cache invalidation after commit.
r1311 f_path):
if repo.alias == 'hg':
from vcs.backends.hg import MercurialInMemoryChangeset as IMC
elif repo.alias == 'git':
from vcs.backends.git import GitInMemoryChangeset as IMC
# decoding here will force that we have proper encoded values
# in any other case this will throw exceptions and deny commit
content = content.encode('utf8')
message = message.encode('utf8')
path = f_path.encode('utf8')
author = author.encode('utf8')
m = IMC(repo)
m.change(FileNode(path, content))
logged local commit with special action via action_logger,
r1312 tip = m.commit(message=message,
moved out commit into scm model, and added cache invalidation after commit.
r1311 author=author,
parents=[cs], branch=cs.branch)
logged local commit with special action via action_logger,
r1312 new_cs = tip.short_id
action = 'push_local:%s' % new_cs
action_logger(user, action, repo_name)
moved out commit into scm model, and added cache invalidation after commit.
r1311 self.mark_for_invalidation(repo_name)
Implemented fancier top menu for logged and anonymous users...
r784 def get_unread_journal(self):
return self.sa.query(UserLog).count()
#50 on point cache invalidation changes....
r692 def _should_invalidate(self, repo_name):
docs updates
r811 """Looks up database for invalidation signals for this repo_name
source code cleanup: remove trailing white space, normalize file endings
r1203
#50 on point cache invalidation changes....
r692 :param repo_name:
"""
ret = self.sa.query(CacheInvalidation)\
.filter(CacheInvalidation.cache_key == repo_name)\
.filter(CacheInvalidation.cache_active == False)\
.scalar()
return ret
def _mark_invalidated(self, cache_key):
source code cleanup: remove trailing white space, normalize file endings
r1203 """ Marks all occurrences of cache to invalidation as already
another major code rafactor, reimplemented (almost from scratch)...
r1038 invalidated
source code cleanup: remove trailing white space, normalize file endings
r1203
docs updates
r811 :param cache_key:
#50 on point cache invalidation changes....
r692 """
docs updates
r811
#50 on point cache invalidation changes....
r692 if cache_key:
log.debug('marking %s as already invalidated', cache_key)
try:
cache_key.cache_active = True
self.sa.add(cache_key)
self.sa.commit()
code cleanups
r758 except (DatabaseError,):
#50 on point cache invalidation changes....
r692 log.error(traceback.format_exc())
self.sa.rollback()