Parent: [0f528b] (diff)

Child: [27f908] (diff)

Download this file

hg.py    135 lines (118 with data), 4.4 kB

  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
import os
import json
import shutil
import logging
import pylons
from pymongo import bson
from dateutil.parser import parse as parse_dt
from forgescm import model as M
from .command import Command
log = logging.getLogger(__name__)
class init(Command):
base='hg init'
class clone(Command):
base='hg clone'
class scm_log(Command):
base='hg log'
def setup_commit_hook(repo_dir, plugin_id):
text = '''[hooks]
incoming.notify_forge = python:forgescm.lib.hg.incoming_hook
[notify_forge]
repository = %s
config = %s
''' % (plugin_id, pylons.config.__file__)
fn = os.path.join(repo_dir, '.hg/hgrc')
with open(fn, 'a') as fp:
fp.write(text)
def incoming_hook(ui, repo, node, **kwargs):
ini_file = ui.config('notify_forge', 'config')
repo_id = ui.config('notify_forge', 'repository')
from pyforge.command import SendMessageCommand
cmd = SendMessageCommand('sendmsg')
msg = dict(hash=node)
cmd.parse_args(['-c', repo_id, ini_file,
'react', 'scm.hg.refresh_commit',
json.dumps(msg)])
cmd.command()
class LogParser(object):
def __init__(self, repo_id):
self.repo_id = repo_id
self.result = []
def feed(self, line_iter):
try:
cur_line = line_iter.next()
while True:
if cur_line.startswith('changeset:'):
cur_line = self.parse_header(cur_line, line_iter)
elif cur_line.startswith('diff --git'):
cur_line = self.parse_diff(cur_line, line_iter)
elif cur_line.strip():
log.error('Unexpected line %r', cur_line)
cur_line = line_iter.next()
else:
cur_line = line_iter.next()
except StopIteration:
pass
return self.result
def parse_header(self, cur_line, line_iter):
hdr, rev, hash = cur_line.split(':')
rev = rev.strip()
hash = hash.strip()
# log.info('Parsing changeset %s:%s', rev, hash)
r = M.Commit(repository_id=self.repo_id,
rev=int(rev),
hash=hash)
while cur_line != '\n':
cur_line = line_iter.next()
if cur_line == '\n': break
try:
cmd, rest = cur_line.split(':', 1)
except ValueError:
if r.summary:
r.summary += cur_line
else:
r.summary = cur_line
continue
result = self.parse_line(rest)
if cur_line.startswith('tag:'):
r.tags.append(result)
elif cur_line.startswith('parent:'):
if not result.startswith('-1:0000'):
r.parents.append(result)
elif cur_line.startswith('user:'):
r.user = result
elif cur_line.startswith('branch:'):
r.branch = result
elif cur_line.startswith('date:'):
r.date = parse_dt(result)
elif cur_line.startswith('summary:'):
r.summary = result
elif cur_line.startswith('description:'):
r.summary = result
elif (cur_line.startswith('extra:')
and 'convert_revision=svn:' in cur_line):
r.rev = int(cur_line.split('@')[-1])
elif cur_line != '\n':
log.debug('Unknown header: %r', cur_line)
if self.result and not self.result[-1].parents:
self.result[-1].parents = [ r.hash ]
self.result.append(r)
if cur_line == '\n':
cur_line = line_iter.next()
return cur_line
def parse_line(self, rest):
return rest.lstrip()
def parse_diff(self, cur_line, line_iter):
cmdline = cur_line.split(' ')
r = M.Patch(repository_id=self.result[-1].repository_id,
commit_id=self.result[-1]._id,
filename=cmdline[2][2:])
text_lines = []
while cur_line != '\n':
cur_line = line_iter.next()
if cur_line.startswith('diff'): break
if cur_line != '\n': text_lines.append(cur_line)
r.patch_text = bson.Binary(''.join(text_lines))
if cur_line == '\n':
cur_line = line_iter.next()
return cur_line