daemon.py
248 lines
| 8.3 KiB
| text/x-python
|
PythonLexer
r547 | #!/usr/bin/env python | |||
# encoding: utf-8 | ||||
r549 | # whoosh indexer daemon for rhodecode | |||
r547 | # Copyright (C) 2009-2010 Marcin Kuzminski <marcin@python-works.com> | |||
# | ||||
# 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 Jan 26, 2010 | ||||
@author: marcink | ||||
A deamon will read from task table and run tasks | ||||
""" | ||||
import sys | ||||
import os | ||||
from os.path import dirname as dn | ||||
from os.path import join as jn | ||||
#to get the rhodecode import | ||||
project_path = dn(dn(dn(dn(os.path.realpath(__file__))))) | ||||
sys.path.append(project_path) | ||||
from rhodecode.lib.pidlock import LockHeld, DaemonLock | ||||
from rhodecode.model.hg_model import HgModel | ||||
from rhodecode.lib.helpers import safe_unicode | ||||
from whoosh.index import create_in, open_dir | ||||
from shutil import rmtree | ||||
from rhodecode.lib.indexers import INDEX_EXTENSIONS, IDX_LOCATION, SCHEMA, IDX_NAME | ||||
r560 | from time import mktime | |||
r567 | from vcs.exceptions import ChangesetError, RepositoryError | |||
r560 | ||||
r547 | import logging | |||
log = logging.getLogger('whooshIndexer') | ||||
# create logger | ||||
log.setLevel(logging.DEBUG) | ||||
log.propagate = False | ||||
# create console handler and set level to debug | ||||
ch = logging.StreamHandler() | ||||
ch.setLevel(logging.DEBUG) | ||||
# create formatter | ||||
formatter = logging.Formatter("%(asctime)s - %(name)s - %(levelname)s - %(message)s") | ||||
# add formatter to ch | ||||
ch.setFormatter(formatter) | ||||
# add ch to logger | ||||
log.addHandler(ch) | ||||
def scan_paths(root_location): | ||||
return HgModel.repo_scan('/', root_location, None, True) | ||||
class WhooshIndexingDaemon(object): | ||||
r560 | """ | |||
Deamon for atomic jobs | ||||
""" | ||||
r547 | ||||
def __init__(self, indexname='HG_INDEX', repo_location=None): | ||||
self.indexname = indexname | ||||
self.repo_location = repo_location | ||||
r561 | self.repo_paths = scan_paths(self.repo_location) | |||
r547 | self.initial = False | |||
if not os.path.isdir(IDX_LOCATION): | ||||
os.mkdir(IDX_LOCATION) | ||||
log.info('Cannot run incremental index since it does not' | ||||
' yet exist running full build') | ||||
self.initial = True | ||||
r560 | ||||
r561 | def get_paths(self, repo): | |||
r560 | """ | |||
recursive walk in root dir and return a set of all path in that dir | ||||
based on repository walk function | ||||
""" | ||||
r547 | index_paths_ = set() | |||
r567 | try: | |||
r627 | tip = repo.get_changeset() | |||
r603 | for topnode, dirs, files in tip.walk('/'): | |||
r547 | for f in files: | |||
r561 | index_paths_.add(jn(repo.path, f.path)) | |||
r567 | for dir in dirs: | |||
for f in files: | ||||
index_paths_.add(jn(repo.path, f.path)) | ||||
except RepositoryError: | ||||
pass | ||||
r560 | return index_paths_ | |||
r561 | ||||
def get_node(self, repo, path): | ||||
n_path = path[len(repo.path) + 1:] | ||||
node = repo.get_changeset().get_node(n_path) | ||||
return node | ||||
def get_node_mtime(self, node): | ||||
return mktime(node.last_changeset.date.timetuple()) | ||||
r547 | def add_doc(self, writer, path, repo): | |||
"""Adding doc to writer""" | ||||
r561 | node = self.get_node(repo, path) | |||
r560 | ||||
#we just index the content of chosen files | ||||
if node.extension in INDEX_EXTENSIONS: | ||||
r547 | log.debug(' >> %s [WITH CONTENT]' % path) | |||
r560 | u_content = node.content | |||
r547 | else: | |||
log.debug(' >> %s' % path) | ||||
#just index file name without it's content | ||||
u_content = u'' | ||||
r560 | writer.add_document(owner=unicode(repo.contact), | |||
repository=safe_unicode(repo.name), | ||||
path=safe_unicode(path), | ||||
content=u_content, | ||||
r561 | modtime=self.get_node_mtime(node), | |||
r560 | extension=node.extension) | |||
r547 | ||||
def build_index(self): | ||||
if os.path.exists(IDX_LOCATION): | ||||
r560 | log.debug('removing previous index') | |||
r547 | rmtree(IDX_LOCATION) | |||
if not os.path.exists(IDX_LOCATION): | ||||
os.mkdir(IDX_LOCATION) | ||||
idx = create_in(IDX_LOCATION, SCHEMA, indexname=IDX_NAME) | ||||
writer = idx.writer() | ||||
r561 | for cnt, repo in enumerate(self.repo_paths.values()): | |||
r547 | log.debug('building index @ %s' % repo.path) | |||
r561 | for idx_path in self.get_paths(repo): | |||
r547 | self.add_doc(writer, idx_path, repo) | |||
r561 | ||||
log.debug('>> COMMITING CHANGES <<') | ||||
r547 | writer.commit(merge=True) | |||
log.debug('>>> FINISHED BUILDING INDEX <<<') | ||||
def update_index(self): | ||||
log.debug('STARTING INCREMENTAL INDEXING UPDATE') | ||||
idx = open_dir(IDX_LOCATION, indexname=self.indexname) | ||||
# The set of all paths in the index | ||||
indexed_paths = set() | ||||
# The set of all paths we need to re-index | ||||
to_index = set() | ||||
reader = idx.reader() | ||||
writer = idx.writer() | ||||
# Loop over the stored fields in the index | ||||
for fields in reader.all_stored_fields(): | ||||
indexed_path = fields['path'] | ||||
indexed_paths.add(indexed_path) | ||||
r561 | ||||
repo = self.repo_paths[fields['repository']] | ||||
try: | ||||
node = self.get_node(repo, indexed_path) | ||||
except ChangesetError: | ||||
r547 | # This file was deleted since it was indexed | |||
log.debug('removing from index %s' % indexed_path) | ||||
writer.delete_by_term('path', indexed_path) | ||||
else: | ||||
r561 | # Check if this file was changed since it was indexed | |||
r547 | indexed_time = fields['modtime'] | |||
r561 | mtime = self.get_node_mtime(node) | |||
r547 | if mtime > indexed_time: | |||
# The file has changed, delete it and add it to the list of | ||||
# files to reindex | ||||
log.debug('adding to reindex list %s' % indexed_path) | ||||
writer.delete_by_term('path', indexed_path) | ||||
to_index.add(indexed_path) | ||||
# Loop over the files in the filesystem | ||||
# Assume we have a function that gathers the filenames of the | ||||
# documents to be indexed | ||||
r561 | for repo in self.repo_paths.values(): | |||
for path in self.get_paths(repo): | ||||
r547 | if path in to_index or path not in indexed_paths: | |||
# This is either a file that's changed, or a new file | ||||
# that wasn't indexed before. So index it! | ||||
self.add_doc(writer, path, repo) | ||||
r561 | log.debug('re indexing %s' % path) | |||
log.debug('>> COMMITING CHANGES <<') | ||||
r547 | writer.commit(merge=True) | |||
r561 | log.debug('>>> FINISHED REBUILDING INDEX <<<') | |||
r547 | ||||
def run(self, full_index=False): | ||||
"""Run daemon""" | ||||
if full_index or self.initial: | ||||
self.build_index() | ||||
else: | ||||
self.update_index() | ||||
if __name__ == "__main__": | ||||
arg = sys.argv[1:] | ||||
if len(arg) != 2: | ||||
sys.stderr.write('Please specify indexing type [full|incremental]' | ||||
'and path to repositories as script args \n') | ||||
sys.exit() | ||||
if arg[0] == 'full': | ||||
full_index = True | ||||
elif arg[0] == 'incremental': | ||||
# False means looking just for changes | ||||
full_index = False | ||||
else: | ||||
sys.stdout.write('Please use [full|incremental]' | ||||
' as script first arg \n') | ||||
sys.exit() | ||||
if not os.path.isdir(arg[1]): | ||||
sys.stderr.write('%s is not a valid path \n' % arg[1]) | ||||
sys.exit() | ||||
else: | ||||
if arg[1].endswith('/'): | ||||
repo_location = arg[1] + '*' | ||||
else: | ||||
repo_location = arg[1] + '/*' | ||||
try: | ||||
l = DaemonLock() | ||||
WhooshIndexingDaemon(repo_location=repo_location)\ | ||||
.run(full_index=full_index) | ||||
l.release() | ||||
reload(logging) | ||||
except LockHeld: | ||||
sys.exit(1) | ||||