# Copyright (c) 2003-2011 LOGILAB S.A. (Paris, FRANCE). # http://www.logilab.fr/ -- mailto:contact@logilab.fr # # 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 2 of the License, or (at your option) any later # version. # # 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., # 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. """helper functions and classes to ease hg revision graph building Based on graphlog's algorithm, with insipration stolen to TortoiseHg revision grapher. """ from cStringIO import StringIO import difflib from mercurial.node import nullrev from mercurial import patch, util, match import hgviewlib # force apply monkeypatches from hgviewlib.util import tounicode, isbfile def diff(repo, ctx1, ctx2=None, files=None): """ Compute the diff of files between 2 changectx """ if ctx2 is None: ctx2 = ctx1.parents()[0] if files is None: m = match.always(repo.root, repo.getcwd()) else: m = match.exact(repo.root, repo.getcwd(), files) # try/except for the sake of hg compatibility (API changes between # 1.0 and 1.1) try: out = StringIO() patch.diff(repo, ctx2.node(), ctx1.node(), match=m, fp=out) diffdata = out.getvalue() except: diffdata = '\n'.join(patch.diff(repo, ctx2.node(), ctx1.node(), match=m)) # XXX how to deal diff encodings? try: diffdata = unicode(diffdata, "utf-8") except UnicodeError: # XXX use a default encoding from config? diffdata = unicode(diffdata, "iso-8859-15", 'ignore') return diffdata def __get_parents(repo, rev, branch=None): """ Return non-null parents of `rev`. If branch is given, only return parents that belongs to names branch `branch` (beware that this is much slower). """ if not branch: if rev is None: return [x.rev() for x in repo.changectx(None).parents() if x] return [x for x in repo.changelog.parentrevs(rev) if x != nullrev] if rev is None: return [x.rev() for x in repo.changectx(None).parents() \ if x and repo.changectx(rev).branch() == branch] return [x for x in repo.changelog.parentrevs(rev) \ if (x != nullrev and repo.changectx(rev).branch() == branch)] def ismerge(ctx): """ Return True if the changecontext ctx is a merge mode (should work with hg 1.0 and 1.2) """ if ctx: return len(ctx.parents()) == 2 and ctx.parents()[1] return False def revision_grapher(repo, start_rev=None, stop_rev=0, branch=None, follow=False): """incremental revision grapher This generator function walks through the revision history from revision start_rev to revision stop_rev (which must be less than or equal to start_rev) and for each revision emits tuples with the following elements: - current revision - column of the current node in the set of ongoing edges - color of the node (?) - lines; a list of (col, next_col, color) indicating the edges between the current row and the next row - parent revisions of current revision If follow is True, only generated the subtree from the start_rev head. If branch is set, only generated the subtree for the given named branch. """ if start_rev is None and repo.status() == ([],)*7: start_rev = len(repo.changelog) assert start_rev is None or start_rev >= stop_rev curr_rev = start_rev revs = [] rev_color = {} nextcolor = 0 while curr_rev is None or curr_rev >= stop_rev: # Compute revs and next_revs. if curr_rev not in revs: if branch: ctx = repo.changectx(curr_rev) if ctx.branch() != branch: if curr_rev is None: curr_rev = len(repo.changelog) else: curr_rev -= 1 yield None continue # New head. if start_rev and follow and curr_rev != start_rev: curr_rev -= 1 continue revs.append(curr_rev) rev_color[curr_rev] = curcolor = nextcolor nextcolor += 1 p_revs = __get_parents(repo, curr_rev, branch) while p_revs: rev0 = p_revs[0] if rev0 < stop_rev or rev0 in rev_color: break rev_color[rev0] = curcolor p_revs = __get_parents(repo, rev0, branch) curcolor = rev_color[curr_rev] rev_index = revs.index(curr_rev) next_revs = revs[:] # Add parents to next_revs. parents = __get_parents(repo, curr_rev, branch) parents_to_add = [] if len(parents) > 1: preferred_color = None else: preferred_color = curcolor for parent in parents: if parent not in next_revs: parents_to_add.append(parent) if parent not in rev_color: if preferred_color: rev_color[parent] = preferred_color preferred_color = None else: rev_color[parent] = nextcolor nextcolor += 1 preferred_color = None # parents_to_add.sort() next_revs[rev_index:rev_index + 1] = parents_to_add lines = [] for i, rev in enumerate(revs): if rev in next_revs: color = rev_color[rev] lines.append( (i, next_revs.index(rev), color) ) elif rev == curr_rev: for parent in parents: color = rev_color[parent] lines.append( (i, next_revs.index(parent), color) ) yield (curr_rev, rev_index, curcolor, lines, parents) revs = next_revs if curr_rev is None: curr_rev = len(repo.changelog) else: curr_rev -= 1 def filelog_grapher(repo, path): ''' Graph the ancestry of a single file (log). Deletions show up as breaks in the graph. ''' filerev = len(repo.file(path)) - 1 fctx = repo.filectx(path, fileid=filerev) rev = fctx.rev() flog = fctx.filelog() heads = [repo.filectx(path, fileid=flog.rev(x)).rev() for x in flog.heads()] assert rev in heads heads.remove(rev) revs = [] rev_color = {} nextcolor = 0 _paths = {} while rev >= 0: # Compute revs and next_revs if rev not in revs: revs.append(rev) rev_color[rev] = nextcolor ; nextcolor += 1 curcolor = rev_color[rev] index = revs.index(rev) next_revs = revs[:] # Add parents to next_revs fctx = repo.filectx(_paths.get(rev, path), changeid=rev) for pfctx in fctx.parents(): _paths[pfctx.rev()] = pfctx.path() parents = [pfctx.rev() for pfctx in fctx.parents()]# if f.path() == path] parents_to_add = [] for parent in parents: if parent not in next_revs: parents_to_add.append(parent) if len(parents) > 1: rev_color[parent] = nextcolor ; nextcolor += 1 else: rev_color[parent] = curcolor parents_to_add.sort() next_revs[index:index + 1] = parents_to_add lines = [] for i, nrev in enumerate(revs): if nrev in next_revs: color = rev_color[nrev] lines.append( (i, next_revs.index(nrev), color) ) elif nrev == rev: for parent in parents: color = rev_color[parent] lines.append( (i, next_revs.index(parent), color) ) pcrevs = [pfc.rev() for pfc in fctx.parents()] yield (fctx.rev(), index, curcolor, lines, pcrevs, _paths.get(fctx.rev(), path)) revs = next_revs if revs: rev = max(revs) else: rev = -1 if heads and rev <= heads[-1]: rev = heads.pop() class GraphNode(object): """ Simple class to encapsulate e hg node in the revision graph. Does nothing but declaring attributes. """ def __init__(self, rev, xposition, color, lines, parents, ncols=None, extra=None): self.rev = rev self.x = xposition self.color = color if ncols is None: ncols = len(lines) self.cols = ncols self.parents = parents self.bottomlines = lines self.toplines = [] self.extra = extra class Graph(object): """ Graph object to ease hg repo navigation. The Graph object instanciate a `revision_grapher` generator, and provide a `fill` method to build the graph progressively. """ #@timeit def __init__(self, repo, grapher, maxfilesize=100000): self.maxfilesize = maxfilesize self.repo = repo self.maxlog = len(self.repo.changelog) self.grapher = grapher self.nodes = [] self.nodesdict = {} self.max_cols = 0 def build_nodes(self, nnodes=None, rev=None): """ Build up to `nnodes` more nodes in our graph, or build as many nodes required to reach `rev`. If both rev and nnodes are set, build as many nodes as required to reach rev plus nnodes more. """ if self.grapher is None: return False stopped = False mcol = [self.max_cols] for vnext in self.grapher: if vnext is None: continue nrev, xpos, color, lines, parents = vnext[:5] if nrev >= self.maxlog: continue gnode = GraphNode(nrev, xpos, color, lines, parents, extra=vnext[5:]) if self.nodes: gnode.toplines = self.nodes[-1].bottomlines self.nodes.append(gnode) self.nodesdict[nrev] = gnode mcol.append(gnode.cols) if rev is not None and nrev <= rev: rev = None # we reached rev, switching to nnode counter if rev is None: if nnodes is not None: nnodes -= 1 if not nnodes: break else: break else: self.grapher = None stopped = True self.max_cols = max(mcol) return not stopped def isfilled(self): return self.grapher is None def fill(self, step=100): """ Return a generator that fills the graph by bursts of `step` more nodes at each iteration. """ while self.build_nodes(step): yield len(self) yield len(self) def __getitem__(self, idx): if isinstance(idx, slice): # XXX TODO: ensure nodes are built return self.nodes.__getitem__(idx) if idx >= len(self.nodes): # build as many graph nodes as required to answer the # requested idx self.build_nodes(idx) if idx > len(self): return self.nodes[-1] return self.nodes[idx] def __len__(self): # len(graph) is the number of actually built graph nodes return len(self.nodes) def index(self, rev): if len(self) == 0: # graph is empty, let's build some nodes self.build_nodes(10) if rev is not None and rev < self.nodes[-1].rev: self.build_nodes(self.nodes[-1].rev - rev) if rev in self.nodesdict: return self.nodes.index(self.nodesdict[rev]) return -1 def fileflags(self, filename, rev): """ Return a couple of flags ('=', '+', '-' or '?') depending on the nature of the diff for filename between rev and its parents. """ ctx = self.repo.changectx(rev) flags = [] for p in ctx.parents(): changes = self.repo.status(p.node(), ctx.node())[:5] # changes = modified, added, removed, deleted, unknown for flag, lst in zip(["=", "+", "-", "-", "?"], changes): if filename in lst: if flag == "+": renamed = ctx.filectx(filename).renamed() if renamed: flags.append(renamed) break flags.append(flag) break else: flags.append('') return flags def fileflag(self, filename, rev): """ Return a flag (see fileflags) between rev and its first parent """ return self.fileflags(filename, rev)[0] def filename(self, rev): return self.nodesdict[rev].extra[0] def filedata(self, filename, rev, mode='diff'): """XXX written under dubious encoding assumptions """ # XXX This really begins to be a dirty mess... data = "" flag = self.fileflag(filename, rev) ctx = self.repo.changectx(rev) try: fctx = ctx.filectx(filename) except LookupError: fctx = None # may happen for renamed files? if isbfile(filename): data = "[bfile]\n" if fctx: data = fctx.data() data += "footprint: %s\n" % data return "+", data if flag not in ('-', '?'): if fctx is None:# or fctx.node() is None: return '', None if fctx.size() > self.maxfilesize: data = "file too big" return flag, data if flag == "+" or mode == 'file': flag = '+' # return the whole file data = fctx.data() if util.binary(data): data = "binary file" else: # tries to convert to unicode data = tounicode(data) elif flag == "=" or isinstance(mode, int): flag = "=" if isinstance(mode, int): parentctx = self.repo.changectx(mode) else: parent = self.fileparent(filename, rev) parentctx = self.repo.changectx(parent) # return the diff but the 3 first lines data = diff(self.repo, ctx, parentctx, files=[filename]) data = u'\n'.join(data.splitlines()[3:]) elif flag == '': data = '' else: # file renamed oldname, node = flag newdata = fctx.data().splitlines() olddata = self.repo.filectx(oldname, fileid=node) olddata = olddata.data().splitlines() data = list(difflib.unified_diff(olddata, newdata, oldname, filename))[2:] if data: flag = "=" else: data = newdata flag = "+" data = u'\n'.join(tounicode(elt) for elt in data) return flag, data def fileparent(self, filename, rev): if rev is not None: node = self.repo.changelog.node(rev) else: node = self.repo.changectx(rev).node() for parent in self.nodesdict[rev].parents: pnode = self.repo.changelog.node(parent) changes = self.repo.status(pnode, node)[:5] allchanges = [] [allchanges.extend(e) for e in changes] if filename in allchanges: return parent return None if __name__ == "__main__": # pylint: disable-msg=C0103 import sys from mercurial import ui, hg u = ui.ui() r = hg.repository(u, sys.argv[1]) if len(sys.argv) == 3: rg = filelog_grapher(r, sys.argv[2]) else: rg = revision_grapher(r) g = Graph(r, rg)