contrib / remote-helpers / git-remote-bzron commit submodule: don't access the .gitmodules cache entry after removing it (bc8d6b9)
   1#!/usr/bin/env python
   2#
   3# Copyright (c) 2012 Felipe Contreras
   4#
   5
   6#
   7# Just copy to your ~/bin, or anywhere in your $PATH.
   8# Then you can clone with:
   9# % git clone bzr::/path/to/bzr/repo/or/url
  10#
  11# For example:
  12# % git clone bzr::$HOME/myrepo
  13# or
  14# % git clone bzr::lp:myrepo
  15#
  16# If you want to specify which branches you want to track (per repo):
  17# % git config remote.origin.bzr-branches 'trunk, devel, test'
  18#
  19# Where 'origin' is the name of the repository you want to specify the
  20# branches.
  21#
  22
  23import sys
  24
  25import bzrlib
  26if hasattr(bzrlib, "initialize"):
  27    bzrlib.initialize()
  28
  29import bzrlib.plugin
  30bzrlib.plugin.load_plugins()
  31
  32import bzrlib.generate_ids
  33import bzrlib.transport
  34import bzrlib.errors
  35import bzrlib.ui
  36import bzrlib.urlutils
  37import bzrlib.branch
  38
  39import sys
  40import os
  41import json
  42import re
  43import StringIO
  44import atexit, shutil, hashlib, urlparse, subprocess
  45
  46NAME_RE = re.compile('^([^<>]+)')
  47AUTHOR_RE = re.compile('^([^<>]+?)? ?<([^<>]*)>$')
  48EMAIL_RE = re.compile('^([^<>]+[^ \\\t<>])?\\b(?:[ \\t<>]*?)\\b([^ \\t<>]+@[^ \\t<>]+)')
  49RAW_AUTHOR_RE = re.compile('^(\w+) (.+)? <(.*)> (\d+) ([+-]\d+)')
  50
  51def die(msg, *args):
  52    sys.stderr.write('ERROR: %s\n' % (msg % args))
  53    sys.exit(1)
  54
  55def warn(msg, *args):
  56    sys.stderr.write('WARNING: %s\n' % (msg % args))
  57
  58def gittz(tz):
  59    return '%+03d%02d' % (tz / 3600, tz % 3600 / 60)
  60
  61def get_config(config):
  62    cmd = ['git', 'config', '--get', config]
  63    process = subprocess.Popen(cmd, stdout=subprocess.PIPE)
  64    output, _ = process.communicate()
  65    return output
  66
  67class Marks:
  68
  69    def __init__(self, path):
  70        self.path = path
  71        self.tips = {}
  72        self.marks = {}
  73        self.rev_marks = {}
  74        self.last_mark = 0
  75        self.load()
  76
  77    def load(self):
  78        if not os.path.exists(self.path):
  79            return
  80
  81        tmp = json.load(open(self.path))
  82        self.tips = tmp['tips']
  83        self.marks = tmp['marks']
  84        self.last_mark = tmp['last-mark']
  85
  86        for rev, mark in self.marks.iteritems():
  87            self.rev_marks[mark] = rev
  88
  89    def dict(self):
  90        return { 'tips': self.tips, 'marks': self.marks, 'last-mark' : self.last_mark }
  91
  92    def store(self):
  93        json.dump(self.dict(), open(self.path, 'w'))
  94
  95    def __str__(self):
  96        return str(self.dict())
  97
  98    def from_rev(self, rev):
  99        return self.marks[rev]
 100
 101    def to_rev(self, mark):
 102        return str(self.rev_marks[mark])
 103
 104    def next_mark(self):
 105        self.last_mark += 1
 106        return self.last_mark
 107
 108    def get_mark(self, rev):
 109        self.last_mark += 1
 110        self.marks[rev] = self.last_mark
 111        return self.last_mark
 112
 113    def is_marked(self, rev):
 114        return rev in self.marks
 115
 116    def new_mark(self, rev, mark):
 117        self.marks[rev] = mark
 118        self.rev_marks[mark] = rev
 119        self.last_mark = mark
 120
 121    def get_tip(self, branch):
 122        try:
 123            return str(self.tips[branch])
 124        except KeyError:
 125            return None
 126
 127    def set_tip(self, branch, tip):
 128        self.tips[branch] = tip
 129
 130class Parser:
 131
 132    def __init__(self, repo):
 133        self.repo = repo
 134        self.line = self.get_line()
 135
 136    def get_line(self):
 137        return sys.stdin.readline().strip()
 138
 139    def __getitem__(self, i):
 140        return self.line.split()[i]
 141
 142    def check(self, word):
 143        return self.line.startswith(word)
 144
 145    def each_block(self, separator):
 146        while self.line != separator:
 147            yield self.line
 148            self.line = self.get_line()
 149
 150    def __iter__(self):
 151        return self.each_block('')
 152
 153    def next(self):
 154        self.line = self.get_line()
 155        if self.line == 'done':
 156            self.line = None
 157
 158    def get_mark(self):
 159        i = self.line.index(':') + 1
 160        return int(self.line[i:])
 161
 162    def get_data(self):
 163        if not self.check('data'):
 164            return None
 165        i = self.line.index(' ') + 1
 166        size = int(self.line[i:])
 167        return sys.stdin.read(size)
 168
 169    def get_author(self):
 170        m = RAW_AUTHOR_RE.match(self.line)
 171        if not m:
 172            return None
 173        _, name, email, date, tz = m.groups()
 174        name = name.decode('utf-8')
 175        committer = '%s <%s>' % (name, email)
 176        tz = int(tz)
 177        tz = ((tz / 100) * 3600) + ((tz % 100) * 60)
 178        return (committer, int(date), tz)
 179
 180def rev_to_mark(rev):
 181    return marks.from_rev(rev)
 182
 183def mark_to_rev(mark):
 184    return marks.to_rev(mark)
 185
 186def fixup_user(user):
 187    name = mail = None
 188    user = user.replace('"', '')
 189    m = AUTHOR_RE.match(user)
 190    if m:
 191        name = m.group(1)
 192        mail = m.group(2).strip()
 193    else:
 194        m = EMAIL_RE.match(user)
 195        if m:
 196            name = m.group(1)
 197            mail = m.group(2)
 198        else:
 199            m = NAME_RE.match(user)
 200            if m:
 201                name = m.group(1).strip()
 202
 203    if not name:
 204        name = 'unknown'
 205    if not mail:
 206        mail = 'Unknown'
 207
 208    return '%s <%s>' % (name, mail)
 209
 210def get_filechanges(cur, prev):
 211    modified = {}
 212    removed = {}
 213
 214    changes = cur.changes_from(prev)
 215
 216    def u(s):
 217        return s.encode('utf-8')
 218
 219    for path, fid, kind in changes.added:
 220        modified[u(path)] = fid
 221    for path, fid, kind in changes.removed:
 222        removed[u(path)] = None
 223    for path, fid, kind, mod, _ in changes.modified:
 224        modified[u(path)] = fid
 225    for oldpath, newpath, fid, kind, mod, _ in changes.renamed:
 226        removed[u(oldpath)] = None
 227        if kind == 'directory':
 228            lst = cur.list_files(from_dir=newpath, recursive=True)
 229            for path, file_class, kind, fid, entry in lst:
 230                if kind != 'directory':
 231                    modified[u(newpath + '/' + path)] = fid
 232        else:
 233            modified[u(newpath)] = fid
 234
 235    return modified, removed
 236
 237def export_files(tree, files):
 238    final = []
 239    for path, fid in files.iteritems():
 240        kind = tree.kind(fid)
 241
 242        h = tree.get_file_sha1(fid)
 243
 244        if kind == 'symlink':
 245            d = tree.get_symlink_target(fid)
 246            mode = '120000'
 247        elif kind == 'file':
 248
 249            if tree.is_executable(fid):
 250                mode = '100755'
 251            else:
 252                mode = '100644'
 253
 254            # is the blob already exported?
 255            if h in filenodes:
 256                mark = filenodes[h]
 257                final.append((mode, mark, path))
 258                continue
 259
 260            d = tree.get_file_text(fid)
 261        elif kind == 'directory':
 262            continue
 263        else:
 264            die("Unhandled kind '%s' for path '%s'" % (kind, path))
 265
 266        mark = marks.next_mark()
 267        filenodes[h] = mark
 268
 269        print "blob"
 270        print "mark :%u" % mark
 271        print "data %d" % len(d)
 272        print d
 273
 274        final.append((mode, mark, path))
 275
 276    return final
 277
 278def export_branch(repo, name):
 279    ref = '%s/heads/%s' % (prefix, name)
 280    tip = marks.get_tip(name)
 281
 282    branch = get_remote_branch(name)
 283    repo = branch.repository
 284
 285    branch.lock_read()
 286    revs = branch.iter_merge_sorted_revisions(None, tip, 'exclude', 'forward')
 287    try:
 288        tip_revno = branch.revision_id_to_revno(tip)
 289        last_revno, _ = branch.last_revision_info()
 290        total = last_revno - tip_revno
 291    except bzrlib.errors.NoSuchRevision:
 292        tip_revno = 0
 293        total = 0
 294
 295    for revid, _, seq, _ in revs:
 296
 297        if marks.is_marked(revid):
 298            continue
 299
 300        rev = repo.get_revision(revid)
 301        revno = seq[0]
 302
 303        parents = rev.parent_ids
 304        time = rev.timestamp
 305        tz = rev.timezone
 306        committer = rev.committer.encode('utf-8')
 307        committer = "%s %u %s" % (fixup_user(committer), time, gittz(tz))
 308        authors = rev.get_apparent_authors()
 309        if authors:
 310            author = authors[0].encode('utf-8')
 311            author = "%s %u %s" % (fixup_user(author), time, gittz(tz))
 312        else:
 313            author = committer
 314        msg = rev.message.encode('utf-8')
 315
 316        msg += '\n'
 317
 318        if len(parents) == 0:
 319            parent = bzrlib.revision.NULL_REVISION
 320        else:
 321            parent = parents[0]
 322
 323        cur_tree = repo.revision_tree(revid)
 324        prev = repo.revision_tree(parent)
 325        modified, removed = get_filechanges(cur_tree, prev)
 326
 327        modified_final = export_files(cur_tree, modified)
 328
 329        if len(parents) == 0:
 330            print 'reset %s' % ref
 331
 332        print "commit %s" % ref
 333        print "mark :%d" % (marks.get_mark(revid))
 334        print "author %s" % (author)
 335        print "committer %s" % (committer)
 336        print "data %d" % (len(msg))
 337        print msg
 338
 339        for i, p in enumerate(parents):
 340            try:
 341                m = rev_to_mark(p)
 342            except KeyError:
 343                # ghost?
 344                continue
 345            if i == 0:
 346                print "from :%s" % m
 347            else:
 348                print "merge :%s" % m
 349
 350        for f in removed:
 351            print "D %s" % (f,)
 352        for f in modified_final:
 353            print "M %s :%u %s" % f
 354        print
 355
 356        if len(seq) > 1:
 357            # let's skip branch revisions from the progress report
 358            continue
 359
 360        progress = (revno - tip_revno)
 361        if (progress % 100 == 0):
 362            if total:
 363                print "progress revision %d '%s' (%d/%d)" % (revno, name, progress, total)
 364            else:
 365                print "progress revision %d '%s' (%d)" % (revno, name, progress)
 366
 367    branch.unlock()
 368
 369    revid = branch.last_revision()
 370
 371    # make sure the ref is updated
 372    print "reset %s" % ref
 373    print "from :%u" % rev_to_mark(revid)
 374    print
 375
 376    marks.set_tip(name, revid)
 377
 378def export_tag(repo, name):
 379    ref = '%s/tags/%s' % (prefix, name)
 380    print "reset %s" % ref
 381    print "from :%u" % rev_to_mark(tags[name])
 382    print
 383
 384def do_import(parser):
 385    repo = parser.repo
 386    path = os.path.join(dirname, 'marks-git')
 387
 388    print "feature done"
 389    if os.path.exists(path):
 390        print "feature import-marks=%s" % path
 391    print "feature export-marks=%s" % path
 392    print "feature force"
 393    sys.stdout.flush()
 394
 395    while parser.check('import'):
 396        ref = parser[1]
 397        if ref.startswith('refs/heads/'):
 398            name = ref[len('refs/heads/'):]
 399            export_branch(repo, name)
 400        if ref.startswith('refs/tags/'):
 401            name = ref[len('refs/tags/'):]
 402            export_tag(repo, name)
 403        parser.next()
 404
 405    print 'done'
 406
 407    sys.stdout.flush()
 408
 409def parse_blob(parser):
 410    parser.next()
 411    mark = parser.get_mark()
 412    parser.next()
 413    data = parser.get_data()
 414    blob_marks[mark] = data
 415    parser.next()
 416
 417class CustomTree():
 418
 419    def __init__(self, branch, revid, parents, files):
 420        self.updates = {}
 421        self.branch = branch
 422
 423        def copy_tree(revid):
 424            files = files_cache[revid] = {}
 425            branch.lock_read()
 426            tree = branch.repository.revision_tree(revid)
 427            try:
 428                for path, entry in tree.iter_entries_by_dir():
 429                    files[path] = [entry.file_id, None]
 430            finally:
 431                branch.unlock()
 432            return files
 433
 434        if len(parents) == 0:
 435            self.base_id = bzrlib.revision.NULL_REVISION
 436            self.base_files = {}
 437        else:
 438            self.base_id = parents[0]
 439            self.base_files = files_cache.get(self.base_id, None)
 440            if not self.base_files:
 441                self.base_files = copy_tree(self.base_id)
 442
 443        self.files = files_cache[revid] = self.base_files.copy()
 444        self.rev_files = {}
 445
 446        for path, data in self.files.iteritems():
 447            fid, mark = data
 448            self.rev_files[fid] = [path, mark]
 449
 450        for path, f in files.iteritems():
 451            fid, mark = self.files.get(path, [None, None])
 452            if not fid:
 453                fid = bzrlib.generate_ids.gen_file_id(path)
 454            f['path'] = path
 455            self.rev_files[fid] = [path, mark]
 456            self.updates[fid] = f
 457
 458    def last_revision(self):
 459        return self.base_id
 460
 461    def iter_changes(self):
 462        changes = []
 463
 464        def get_parent(dirname, basename):
 465            parent_fid, mark = self.base_files.get(dirname, [None, None])
 466            if parent_fid:
 467                return parent_fid
 468            parent_fid, mark = self.files.get(dirname, [None, None])
 469            if parent_fid:
 470                return parent_fid
 471            if basename == '':
 472                return None
 473            fid = bzrlib.generate_ids.gen_file_id(path)
 474            add_entry(fid, dirname, 'directory')
 475            return fid
 476
 477        def add_entry(fid, path, kind, mode=None):
 478            dirname, basename = os.path.split(path)
 479            parent_fid = get_parent(dirname, basename)
 480
 481            executable = False
 482            if mode == '100755':
 483                executable = True
 484            elif mode == '120000':
 485                kind = 'symlink'
 486
 487            change = (fid,
 488                    (None, path),
 489                    True,
 490                    (False, True),
 491                    (None, parent_fid),
 492                    (None, basename),
 493                    (None, kind),
 494                    (None, executable))
 495            self.files[path] = [change[0], None]
 496            changes.append(change)
 497
 498        def update_entry(fid, path, kind, mode=None):
 499            dirname, basename = os.path.split(path)
 500            parent_fid = get_parent(dirname, basename)
 501
 502            executable = False
 503            if mode == '100755':
 504                executable = True
 505            elif mode == '120000':
 506                kind = 'symlink'
 507
 508            change = (fid,
 509                    (path, path),
 510                    True,
 511                    (True, True),
 512                    (None, parent_fid),
 513                    (None, basename),
 514                    (None, kind),
 515                    (None, executable))
 516            self.files[path] = [change[0], None]
 517            changes.append(change)
 518
 519        def remove_entry(fid, path, kind):
 520            dirname, basename = os.path.split(path)
 521            parent_fid = get_parent(dirname, basename)
 522            change = (fid,
 523                    (path, None),
 524                    True,
 525                    (True, False),
 526                    (parent_fid, None),
 527                    (None, None),
 528                    (None, None),
 529                    (None, None))
 530            del self.files[path]
 531            changes.append(change)
 532
 533        for fid, f in self.updates.iteritems():
 534            path = f['path']
 535
 536            if 'deleted' in f:
 537                remove_entry(fid, path, 'file')
 538                continue
 539
 540            if path in self.base_files:
 541                update_entry(fid, path, 'file', f['mode'])
 542            else:
 543                add_entry(fid, path, 'file', f['mode'])
 544
 545            self.files[path][1] = f['mark']
 546            self.rev_files[fid][1] = f['mark']
 547
 548        return changes
 549
 550    def get_content(self, file_id):
 551        path, mark = self.rev_files[file_id]
 552        if mark:
 553            return blob_marks[mark]
 554
 555        # last resort
 556        tree = self.branch.repository.revision_tree(self.base_id)
 557        return tree.get_file_text(file_id)
 558
 559    def get_file_with_stat(self, file_id, path=None):
 560        content = self.get_content(file_id)
 561        return (StringIO.StringIO(content), None)
 562
 563    def get_symlink_target(self, file_id):
 564        return self.get_content(file_id)
 565
 566    def id2path(self, file_id):
 567        path, mark = self.rev_files[file_id]
 568        return path
 569
 570def c_style_unescape(string):
 571    if string[0] == string[-1] == '"':
 572        return string.decode('string-escape')[1:-1]
 573    return string
 574
 575def parse_commit(parser):
 576    parents = []
 577
 578    ref = parser[1]
 579    parser.next()
 580
 581    if ref.startswith('refs/heads/'):
 582        name = ref[len('refs/heads/'):]
 583        branch = get_remote_branch(name)
 584    else:
 585        die('unknown ref')
 586
 587    commit_mark = parser.get_mark()
 588    parser.next()
 589    author = parser.get_author()
 590    parser.next()
 591    committer = parser.get_author()
 592    parser.next()
 593    data = parser.get_data()
 594    parser.next()
 595    if parser.check('from'):
 596        parents.append(parser.get_mark())
 597        parser.next()
 598    while parser.check('merge'):
 599        parents.append(parser.get_mark())
 600        parser.next()
 601
 602    # fast-export adds an extra newline
 603    if data[-1] == '\n':
 604        data = data[:-1]
 605
 606    files = {}
 607
 608    for line in parser:
 609        if parser.check('M'):
 610            t, m, mark_ref, path = line.split(' ', 3)
 611            mark = int(mark_ref[1:])
 612            f = { 'mode' : m, 'mark' : mark }
 613        elif parser.check('D'):
 614            t, path = line.split(' ', 1)
 615            f = { 'deleted' : True }
 616        else:
 617            die('Unknown file command: %s' % line)
 618        path = c_style_unescape(path).decode('utf-8')
 619        files[path] = f
 620
 621    committer, date, tz = committer
 622    parents = [mark_to_rev(p) for p in parents]
 623    revid = bzrlib.generate_ids.gen_revision_id(committer, date)
 624    props = {}
 625    props['branch-nick'] = branch.nick
 626
 627    mtree = CustomTree(branch, revid, parents, files)
 628    changes = mtree.iter_changes()
 629
 630    branch.lock_write()
 631    try:
 632        builder = branch.get_commit_builder(parents, None, date, tz, committer, props, revid)
 633        try:
 634            list(builder.record_iter_changes(mtree, mtree.last_revision(), changes))
 635            builder.finish_inventory()
 636            builder.commit(data.decode('utf-8', 'replace'))
 637        except Exception, e:
 638            builder.abort()
 639            raise
 640    finally:
 641        branch.unlock()
 642
 643    parsed_refs[ref] = revid
 644    marks.new_mark(revid, commit_mark)
 645
 646def parse_reset(parser):
 647    ref = parser[1]
 648    parser.next()
 649
 650    # ugh
 651    if parser.check('commit'):
 652        parse_commit(parser)
 653        return
 654    if not parser.check('from'):
 655        return
 656    from_mark = parser.get_mark()
 657    parser.next()
 658
 659    parsed_refs[ref] = mark_to_rev(from_mark)
 660
 661def do_export(parser):
 662    parser.next()
 663
 664    for line in parser.each_block('done'):
 665        if parser.check('blob'):
 666            parse_blob(parser)
 667        elif parser.check('commit'):
 668            parse_commit(parser)
 669        elif parser.check('reset'):
 670            parse_reset(parser)
 671        elif parser.check('tag'):
 672            pass
 673        elif parser.check('feature'):
 674            pass
 675        else:
 676            die('unhandled export command: %s' % line)
 677
 678    for ref, revid in parsed_refs.iteritems():
 679        if ref.startswith('refs/heads/'):
 680            name = ref[len('refs/heads/'):]
 681            branch = get_remote_branch(name)
 682            branch.generate_revision_history(revid, marks.get_tip(name))
 683
 684            if name in peers:
 685                peer = bzrlib.branch.Branch.open(peers[name],
 686                                                 possible_transports=transports)
 687                try:
 688                    peer.bzrdir.push_branch(branch, revision_id=revid)
 689                except bzrlib.errors.DivergedBranches:
 690                    print "error %s non-fast forward" % ref
 691                    continue
 692
 693            try:
 694                wt = branch.bzrdir.open_workingtree()
 695                wt.update()
 696            except bzrlib.errors.NoWorkingTree:
 697                pass
 698        elif ref.startswith('refs/tags/'):
 699            # TODO: implement tag push
 700            print "error %s pushing tags not supported" % ref
 701            continue
 702        else:
 703            # transport-helper/fast-export bugs
 704            continue
 705
 706        print "ok %s" % ref
 707
 708    print
 709
 710def do_capabilities(parser):
 711    print "import"
 712    print "export"
 713    print "refspec refs/heads/*:%s/heads/*" % prefix
 714    print "refspec refs/tags/*:%s/tags/*" % prefix
 715
 716    path = os.path.join(dirname, 'marks-git')
 717
 718    if os.path.exists(path):
 719        print "*import-marks %s" % path
 720    print "*export-marks %s" % path
 721
 722    print
 723
 724def ref_is_valid(name):
 725    return not True in [c in name for c in '~^: \\']
 726
 727def do_list(parser):
 728    master_branch = None
 729
 730    for name in branches:
 731        if not master_branch:
 732            master_branch = name
 733        print "? refs/heads/%s" % name
 734
 735    branch = get_remote_branch(master_branch)
 736    branch.lock_read()
 737    for tag, revid in branch.tags.get_tag_dict().items():
 738        try:
 739            branch.revision_id_to_dotted_revno(revid)
 740        except bzrlib.errors.NoSuchRevision:
 741            continue
 742        if not ref_is_valid(tag):
 743            continue
 744        print "? refs/tags/%s" % tag
 745        tags[tag] = revid
 746    branch.unlock()
 747
 748    print "@refs/heads/%s HEAD" % master_branch
 749    print
 750
 751def clone(path, remote_branch):
 752    try:
 753        bdir = bzrlib.bzrdir.BzrDir.create(path, possible_transports=transports)
 754    except bzrlib.errors.AlreadyControlDirError:
 755        bdir = bzrlib.bzrdir.BzrDir.open(path, possible_transports=transports)
 756    repo = bdir.find_repository()
 757    repo.fetch(remote_branch.repository)
 758    return remote_branch.sprout(bdir, repository=repo)
 759
 760def get_remote_branch(name):
 761    remote_branch = bzrlib.branch.Branch.open(branches[name],
 762                                              possible_transports=transports)
 763    if isinstance(remote_branch.user_transport, bzrlib.transport.local.LocalTransport):
 764        return remote_branch
 765
 766    branch_path = os.path.join(dirname, 'clone', name)
 767
 768    try:
 769        branch = bzrlib.branch.Branch.open(branch_path,
 770                                           possible_transports=transports)
 771    except bzrlib.errors.NotBranchError:
 772        # clone
 773        branch = clone(branch_path, remote_branch)
 774    else:
 775        # pull
 776        try:
 777            branch.pull(remote_branch, overwrite=True)
 778        except bzrlib.errors.DivergedBranches:
 779            # use remote branch for now
 780            return remote_branch
 781
 782    return branch
 783
 784def find_branches(repo):
 785    transport = repo.bzrdir.root_transport
 786
 787    for fn in transport.iter_files_recursive():
 788        if not fn.endswith('.bzr/branch-format'):
 789            continue
 790
 791        name = subdir = fn[:-len('/.bzr/branch-format')]
 792        name = name if name != '' else 'master'
 793        name = name.replace('/', '+')
 794
 795        try:
 796            cur = transport.clone(subdir)
 797            branch = bzrlib.branch.Branch.open_from_transport(cur)
 798        except bzrlib.errors.NotBranchError:
 799            continue
 800        else:
 801            yield name, branch.base
 802
 803def get_repo(url, alias):
 804    normal_url = bzrlib.urlutils.normalize_url(url)
 805    origin = bzrlib.bzrdir.BzrDir.open(url, possible_transports=transports)
 806    is_local = isinstance(origin.transport, bzrlib.transport.local.LocalTransport)
 807
 808    shared_path = os.path.join(gitdir, 'bzr')
 809    try:
 810        shared_dir = bzrlib.bzrdir.BzrDir.open(shared_path,
 811                                               possible_transports=transports)
 812    except bzrlib.errors.NotBranchError:
 813        shared_dir = bzrlib.bzrdir.BzrDir.create(shared_path,
 814                                                 possible_transports=transports)
 815    try:
 816        shared_repo = shared_dir.open_repository()
 817    except bzrlib.errors.NoRepositoryPresent:
 818        shared_repo = shared_dir.create_repository(shared=True)
 819
 820    if not is_local:
 821        clone_path = os.path.join(dirname, 'clone')
 822        if not os.path.exists(clone_path):
 823            os.mkdir(clone_path)
 824        else:
 825            # check and remove old organization
 826            try:
 827                bdir = bzrlib.bzrdir.BzrDir.open(clone_path,
 828                                                 possible_transports=transports)
 829                bdir.destroy_repository()
 830            except bzrlib.errors.NotBranchError:
 831                pass
 832            except bzrlib.errors.NoRepositoryPresent:
 833                pass
 834
 835    wanted = get_config('remote.%s.bzr-branches' % alias).rstrip().split(', ')
 836    # stupid python
 837    wanted = [e for e in wanted if e]
 838    if not wanted:
 839        wanted = get_config('remote-bzr.branches').rstrip().split(', ')
 840        # stupid python
 841        wanted = [e for e in wanted if e]
 842
 843    if not wanted:
 844        try:
 845            repo = origin.open_repository()
 846            if not repo.user_transport.listable():
 847                # this repository is not usable for us
 848                raise bzrlib.errors.NoRepositoryPresent(repo.bzrdir)
 849        except bzrlib.errors.NoRepositoryPresent:
 850            wanted = ['master']
 851
 852    if wanted:
 853        def list_wanted(url, wanted):
 854            for name in wanted:
 855                subdir = name if name != 'master' else ''
 856                yield name, bzrlib.urlutils.join(url, subdir)
 857
 858        branch_list = list_wanted(url, wanted)
 859    else:
 860        branch_list = find_branches(repo)
 861
 862    for name, url in branch_list:
 863        if not is_local:
 864            peers[name] = url
 865        branches[name] = url
 866
 867    return origin
 868
 869def fix_path(alias, orig_url):
 870    url = urlparse.urlparse(orig_url, 'file')
 871    if url.scheme != 'file' or os.path.isabs(url.path):
 872        return
 873    abs_url = urlparse.urljoin("%s/" % os.getcwd(), orig_url)
 874    cmd = ['git', 'config', 'remote.%s.url' % alias, "bzr::%s" % abs_url]
 875    subprocess.call(cmd)
 876
 877def main(args):
 878    global marks, prefix, gitdir, dirname
 879    global tags, filenodes
 880    global blob_marks
 881    global parsed_refs
 882    global files_cache
 883    global is_tmp
 884    global branches, peers
 885    global transports
 886
 887    alias = args[1]
 888    url = args[2]
 889
 890    tags = {}
 891    filenodes = {}
 892    blob_marks = {}
 893    parsed_refs = {}
 894    files_cache = {}
 895    marks = None
 896    branches = {}
 897    peers = {}
 898    transports = []
 899
 900    if alias[5:] == url:
 901        is_tmp = True
 902        alias = hashlib.sha1(alias).hexdigest()
 903    else:
 904        is_tmp = False
 905
 906    prefix = 'refs/bzr/%s' % alias
 907    gitdir = os.environ['GIT_DIR']
 908    dirname = os.path.join(gitdir, 'bzr', alias)
 909
 910    if not is_tmp:
 911        fix_path(alias, url)
 912
 913    if not os.path.exists(dirname):
 914        os.makedirs(dirname)
 915
 916    if hasattr(bzrlib.ui.ui_factory, 'be_quiet'):
 917        bzrlib.ui.ui_factory.be_quiet(True)
 918
 919    repo = get_repo(url, alias)
 920
 921    marks_path = os.path.join(dirname, 'marks-int')
 922    marks = Marks(marks_path)
 923
 924    parser = Parser(repo)
 925    for line in parser:
 926        if parser.check('capabilities'):
 927            do_capabilities(parser)
 928        elif parser.check('list'):
 929            do_list(parser)
 930        elif parser.check('import'):
 931            do_import(parser)
 932        elif parser.check('export'):
 933            do_export(parser)
 934        else:
 935            die('unhandled command: %s' % line)
 936        sys.stdout.flush()
 937
 938def bye():
 939    if not marks:
 940        return
 941    if not is_tmp:
 942        marks.store()
 943    else:
 944        shutil.rmtree(dirname)
 945
 946atexit.register(bye)
 947sys.exit(main(sys.argv))