~abentley/bzrtools/bzrtools.dev

« back to all changes in this revision

Viewing changes to testdata/mod

  • Committer: Aaron Bentley
  • Date: 2008-10-06 23:24:16 UTC
  • Revision ID: aaron@aaronbentley.com-20081006232416-mbohnubye0z5tiev
Add target_file param

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2004 Aaron Bentley
2
 
# <aaron.bentley@utoronto.ca>
3
 
#
4
 
#    This program is free software; you can redistribute it and/or modify
5
 
#    it under the terms of the GNU General Public License as published by
6
 
#    the Free Software Foundation; either version 2 of the License, or
7
 
#    (at your option) any later version.
8
 
#
9
 
#    This program is distributed in the hope that it will be useful,
10
 
#    but WITHOUT ANY WARRANTY; without even the implied warranty of
11
 
#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12
 
#    GNU General Public License for more details.
13
 
#
14
 
#    You should have received a copy of the GNU General Public License
15
 
#    along with this program; if not, write to the Free Software
16
 
#    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
17
 
 
18
 
import sys
19
 
import arch
20
 
import arch.util
21
 
import arch.arch
22
 
 
23
 
import pylon.errors
24
 
from pylon.errors import *
25
 
from pylon import errors
26
 
from pylon import util
27
 
from pylon import arch_core
28
 
from pylon import arch_compound
29
 
from pylon import ancillary
30
 
from pylon import misc
31
 
from pylon import paths 
32
 
 
33
 
import abacmds
34
 
import cmdutil
35
 
import shutil
36
 
import os
37
 
import options
38
 
import time
39
 
import cmd
40
 
import readline
41
 
import re
42
 
import string
43
 
import terminal
44
 
import email
45
 
import smtplib
46
 
import textwrap
47
 
 
48
 
__docformat__ = "restructuredtext"
49
 
__doc__ = "Implementation of user (sub) commands"
50
 
commands = {}
51
 
 
52
 
def find_command(cmd):
53
 
    """
54
 
    Return an instance of a command type.  Return None if the type isn't
55
 
    registered.
56
 
 
57
 
    :param cmd: the name of the command to look for
58
 
    :type cmd: the type of the command
59
 
    """
60
 
    if commands.has_key(cmd):
61
 
        return commands[cmd]()
62
 
    else:
63
 
        return None
64
 
 
65
 
class BaseCommand:
66
 
    def __call__(self, cmdline):
67
 
        try:
68
 
            self.do_command(cmdline.split())
69
 
        except cmdutil.GetHelp, e:
70
 
            self.help()
71
 
        except Exception, e:
72
 
            print e
73
 
 
74
 
    def get_completer(index):
75
 
        return None
76
 
 
77
 
    def complete(self, args, text):
78
 
        """
79
 
        Returns a list of possible completions for the given text.
80
 
 
81
 
        :param args: The complete list of arguments
82
 
        :type args: List of str
83
 
        :param text: text to complete (may be shorter than args[-1])
84
 
        :type text: str
85
 
        :rtype: list of str
86
 
        """
87
 
        matches = []
88
 
        candidates = None
89
 
 
90
 
        if len(args) > 0: 
91
 
            realtext = args[-1]
92
 
        else:
93
 
            realtext = ""
94
 
 
95
 
        try:
96
 
            parser=self.get_parser()
97
 
            if realtext.startswith('-'):
98
 
                candidates = parser.iter_options()
99
 
            else:
100
 
                (options, parsed_args) = parser.parse_args(args)
101
 
 
102
 
                if len (parsed_args) > 0:
103
 
                    candidates = self.get_completer(parsed_args[-1], len(parsed_args) -1)
104
 
                else:
105
 
                    candidates = self.get_completer("", 0)
106
 
        except:
107
 
            pass
108
 
        if candidates is None:
109
 
            return
110
 
        for candidate in candidates:
111
 
            candidate = str(candidate)
112
 
            if candidate.startswith(realtext):
113
 
                matches.append(candidate[len(realtext)- len(text):])
114
 
        return matches
115
 
 
116
 
 
117
 
class Help(BaseCommand):
118
 
    """
119
 
    Lists commands, prints help messages.
120
 
    """
121
 
    def __init__(self):
122
 
        self.description="Prints help mesages"
123
 
        self.parser = None
124
 
 
125
 
    def do_command(self, cmdargs):
126
 
        """
127
 
        Prints a help message.
128
 
        """
129
 
        options, args = self.get_parser().parse_args(cmdargs)
130
 
        if len(args) > 1:
131
 
            raise cmdutil.GetHelp
132
 
 
133
 
        if options.native or options.suggestions or options.external:
134
 
            native = options.native
135
 
            suggestions = options.suggestions
136
 
            external = options.external
137
 
        else:
138
 
            native = True
139
 
            suggestions = False
140
 
            external = True
141
 
        
142
 
        if len(args) == 0:
143
 
            self.list_commands(native, suggestions, external)
144
 
            return
145
 
        elif len(args) == 1:
146
 
            command_help(args[0])
147
 
            return
148
 
 
149
 
    def help(self):
150
 
        self.get_parser().print_help()
151
 
        print """
152
 
If no command is specified, commands are listed.  If a command is
153
 
specified, help for that command is listed.
154
 
        """
155
 
 
156
 
    def get_parser(self):
157
 
        """
158
 
        Returns the options parser to use for the "revision" command.
159
 
 
160
 
        :rtype: cmdutil.CmdOptionParser
161
 
        """
162
 
        if self.parser is not None:
163
 
            return self.parser
164
 
        parser=cmdutil.CmdOptionParser("fai help [command]")
165
 
        parser.add_option("-n", "--native", action="store_true", 
166
 
                         dest="native", help="Show native commands")
167
 
        parser.add_option("-e", "--external", action="store_true", 
168
 
                         dest="external", help="Show external commands")
169
 
        parser.add_option("-s", "--suggest", action="store_true", 
170
 
                         dest="suggestions", help="Show suggestions")
171
 
        self.parser = parser
172
 
        return parser 
173
 
      
174
 
    def list_commands(self, native=True, suggest=False, external=True):
175
 
        """
176
 
        Lists supported commands.
177
 
 
178
 
        :param native: list native, python-based commands
179
 
        :type native: bool
180
 
        :param external: list external aba-style commands
181
 
        :type external: bool
182
 
        """
183
 
        if native:
184
 
            print "Native Fai commands"
185
 
            keys=commands.keys()
186
 
            keys.sort()
187
 
            for k in keys:
188
 
                space=""
189
 
                for i in range(28-len(k)):
190
 
                    space+=" "
191
 
                print space+k+" : "+commands[k]().description
192
 
            print
193
 
        if suggest:
194
 
            print "Unavailable commands and suggested alternatives"
195
 
            key_list = suggestions.keys()
196
 
            key_list.sort()
197
 
            for key in key_list:
198
 
                print "%28s : %s" % (key, suggestions[key])
199
 
            print
200
 
        if external:
201
 
            fake_aba = abacmds.AbaCmds()
202
 
            if (fake_aba.abadir == ""):
203
 
                return
204
 
            print "External commands"
205
 
            fake_aba.list_commands()
206
 
            print
207
 
        if not suggest:
208
 
            print "Use help --suggest to list alternatives to tla and aba"\
209
 
                " commands."
210
 
        if options.tla_fallthrough and (native or external):
211
 
            print "Fai also supports tla commands."
212
 
 
213
 
def command_help(cmd):
214
 
    """
215
 
    Prints help for a command.
216
 
 
217
 
    :param cmd: The name of the command to print help for
218
 
    :type cmd: str
219
 
    """
220
 
    fake_aba = abacmds.AbaCmds()
221
 
    cmdobj = find_command(cmd)
222
 
    if cmdobj != None:
223
 
        cmdobj.help()
224
 
    elif suggestions.has_key(cmd):
225
 
        print "Not available\n" + suggestions[cmd]
226
 
    else:
227
 
        abacmd = fake_aba.is_command(cmd)
228
 
        if abacmd:
229
 
            abacmd.help()
230
 
        else:
231
 
            print "No help is available for \""+cmd+"\". Maybe try \"tla "+cmd+" -H\"?"
232
 
 
233
 
 
234
 
 
235
 
class Changes(BaseCommand):
236
 
    """
237
 
    the "changes" command: lists differences between trees/revisions:
238
 
    """
239
 
    
240
 
    def __init__(self):
241
 
        self.description="Lists what files have changed in the project tree"
242
 
 
243
 
    def get_completer(self, arg, index):
244
 
        if index > 1:
245
 
            return None
246
 
        try:
247
 
            tree = arch.tree_root()
248
 
        except:
249
 
            tree = None
250
 
        return cmdutil.iter_revision_completions(arg, tree)
251
 
    
252
 
    def parse_commandline(self, cmdline):
253
 
        """
254
 
        Parse commandline arguments.  Raises cmdutil.GetHelp if help is needed.
255
 
        
256
 
        :param cmdline: A list of arguments to parse
257
 
        :rtype: (options, Revision, Revision/WorkingTree)
258
 
        """
259
 
        parser=self.get_parser()
260
 
        (options, args) = parser.parse_args(cmdline)
261
 
        if len(args) > 2:
262
 
            raise cmdutil.GetHelp
263
 
 
264
 
        tree=arch.tree_root()
265
 
        if len(args) == 0:
266
 
            a_spec = ancillary.comp_revision(tree)
267
 
        else:
268
 
            a_spec = cmdutil.determine_revision_tree(tree, args[0])
269
 
        cmdutil.ensure_archive_registered(a_spec.archive)
270
 
        if len(args) == 2:
271
 
            b_spec = cmdutil.determine_revision_tree(tree, args[1])
272
 
            cmdutil.ensure_archive_registered(b_spec.archive)
273
 
        else:
274
 
            b_spec=tree
275
 
        return options, a_spec, b_spec
276
 
 
277
 
    def do_command(self, cmdargs):
278
 
        """
279
 
        Master function that perfoms the "changes" command.
280
 
        """
281
 
        try:
282
 
            options, a_spec, b_spec = self.parse_commandline(cmdargs);
283
 
        except cmdutil.CantDetermineRevision, e:
284
 
            print e
285
 
            return
286
 
        except arch.errors.TreeRootError, e:
287
 
            print e
288
 
            return
289
 
        if options.changeset:
290
 
            changeset=options.changeset
291
 
            tmpdir = None
292
 
        else:
293
 
            tmpdir=util.tmpdir()
294
 
            changeset=tmpdir+"/changeset"
295
 
        try:
296
 
            delta=arch.iter_delta(a_spec, b_spec, changeset)
297
 
            try:
298
 
                for line in delta:
299
 
                    if cmdutil.chattermatch(line, "changeset:"):
300
 
                        pass
301
 
                    else:
302
 
                        cmdutil.colorize(line, options.suppress_chatter)
303
 
            except arch.util.ExecProblem, e:
304
 
                if e.proc.error and e.proc.error.startswith(
305
 
                    "missing explicit id for file"):
306
 
                    raise MissingID(e)
307
 
                else:
308
 
                    raise
309
 
            status=delta.status
310
 
            if status > 1:
311
 
                return
312
 
            if (options.perform_diff):
313
 
                chan = arch_compound.ChangesetMunger(changeset)
314
 
                chan.read_indices()
315
 
                if options.diffopts is not None:
316
 
                    if isinstance(b_spec, arch.Revision):
317
 
                        b_dir = b_spec.library_find()
318
 
                    else:
319
 
                        b_dir = b_spec
320
 
                    a_dir = a_spec.library_find()
321
 
                    diffopts = options.diffopts.split()
322
 
                    cmdutil.show_custom_diffs(chan, diffopts, a_dir, b_dir)
323
 
                else:
324
 
                    cmdutil.show_diffs(delta.changeset)
325
 
        finally:
326
 
            if tmpdir and (os.access(tmpdir, os.X_OK)):
327
 
                shutil.rmtree(tmpdir)
328
 
 
329
 
    def get_parser(self):
330
 
        """
331
 
        Returns the options parser to use for the "changes" command.
332
 
 
333
 
        :rtype: cmdutil.CmdOptionParser
334
 
        """
335
 
        parser=cmdutil.CmdOptionParser("fai changes [options] [revision]"
336
 
                                       " [revision]")
337
 
        parser.add_option("-d", "--diff", action="store_true", 
338
 
                          dest="perform_diff", default=False, 
339
 
                          help="Show diffs in summary")
340
 
        parser.add_option("-c", "--changeset", dest="changeset", 
341
 
                          help="Store a changeset in the given directory", 
342
 
                          metavar="DIRECTORY")
343
 
        parser.add_option("-s", "--silent", action="store_true", 
344
 
                          dest="suppress_chatter", default=False, 
345
 
                          help="Suppress chatter messages")
346
 
        parser.add_option("--diffopts", dest="diffopts", 
347
 
                          help="Use the specified diff options", 
348
 
                          metavar="OPTIONS")
349
 
 
350
 
        return parser
351
 
 
352
 
    def help(self, parser=None):
353
 
        """
354
 
        Prints a help message.
355
 
 
356
 
        :param parser: If supplied, the parser to use for generating help.  If \
357
 
        not supplied, it is retrieved.
358
 
        :type parser: cmdutil.CmdOptionParser
359
 
        """
360
 
        if parser is None:
361
 
            parser=self.get_parser()
362
 
        parser.print_help()
363
 
        print """
364
 
Performs source-tree comparisons
365
 
 
366
 
If no revision is specified, the current project tree is compared to the
367
 
last-committed revision.  If one revision is specified, the current project
368
 
tree is compared to that revision.  If two revisions are specified, they are
369
 
compared to each other.
370
 
        """
371
 
        help_tree_spec() 
372
 
        return
373
 
 
374
 
 
375
 
class ApplyChanges(BaseCommand):
376
 
    """
377
 
    Apply differences between two revisions to a tree
378
 
    """
379
 
    
380
 
    def __init__(self):
381
 
        self.description="Applies changes to a project tree"
382
 
    
383
 
    def get_completer(self, arg, index):
384
 
        if index > 1:
385
 
            return None
386
 
        try:
387
 
            tree = arch.tree_root()
388
 
        except:
389
 
            tree = None
390
 
        return cmdutil.iter_revision_completions(arg, tree)
391
 
 
392
 
    def parse_commandline(self, cmdline, tree):
393
 
        """
394
 
        Parse commandline arguments.  Raises cmdutil.GetHelp if help is needed.
395
 
        
396
 
        :param cmdline: A list of arguments to parse
397
 
        :rtype: (options, Revision, Revision/WorkingTree)
398
 
        """
399
 
        parser=self.get_parser()
400
 
        (options, args) = parser.parse_args(cmdline)
401
 
        if len(args) != 2:
402
 
            raise cmdutil.GetHelp
403
 
 
404
 
        a_spec = cmdutil.determine_revision_tree(tree, args[0])
405
 
        cmdutil.ensure_archive_registered(a_spec.archive)
406
 
        b_spec = cmdutil.determine_revision_tree(tree, args[1])
407
 
        cmdutil.ensure_archive_registered(b_spec.archive)
408
 
        return options, a_spec, b_spec
409
 
 
410
 
    def do_command(self, cmdargs):
411
 
        """
412
 
        Master function that performs "apply-changes".
413
 
        """
414
 
        try:
415
 
            tree = arch.tree_root()
416
 
            options, a_spec, b_spec = self.parse_commandline(cmdargs, tree);
417
 
        except cmdutil.CantDetermineRevision, e:
418
 
            print e
419
 
            return
420
 
        except arch.errors.TreeRootError, e:
421
 
            print e
422
 
            return
423
 
        delta=cmdutil.apply_delta(a_spec, b_spec, tree)
424
 
        for line in cmdutil.iter_apply_delta_filter(delta):
425
 
            cmdutil.colorize(line, options.suppress_chatter)
426
 
 
427
 
    def get_parser(self):
428
 
        """
429
 
        Returns the options parser to use for the "apply-changes" command.
430
 
 
431
 
        :rtype: cmdutil.CmdOptionParser
432
 
        """
433
 
        parser=cmdutil.CmdOptionParser("fai apply-changes [options] revision"
434
 
                                       " revision")
435
 
        parser.add_option("-d", "--diff", action="store_true", 
436
 
                          dest="perform_diff", default=False, 
437
 
                          help="Show diffs in summary")
438
 
        parser.add_option("-c", "--changeset", dest="changeset", 
439
 
                          help="Store a changeset in the given directory", 
440
 
                          metavar="DIRECTORY")
441
 
        parser.add_option("-s", "--silent", action="store_true", 
442
 
                          dest="suppress_chatter", default=False, 
443
 
                          help="Suppress chatter messages")
444
 
        return parser
445
 
 
446
 
    def help(self, parser=None):
447
 
        """
448
 
        Prints a help message.
449
 
 
450
 
        :param parser: If supplied, the parser to use for generating help.  If \
451
 
        not supplied, it is retrieved.
452
 
        :type parser: cmdutil.CmdOptionParser
453
 
        """
454
 
        if parser is None:
455
 
            parser=self.get_parser()
456
 
        parser.print_help()
457
 
        print """
458
 
Applies changes to a project tree
459
 
 
460
 
Compares two revisions and applies the difference between them to the current
461
 
tree.
462
 
        """
463
 
        help_tree_spec() 
464
 
        return
465
 
 
466
 
class Update(BaseCommand):
467
 
    """
468
 
    Updates a project tree to a given revision, preserving un-committed hanges. 
469
 
    """
470
 
    
471
 
    def __init__(self):
472
 
        self.description="Apply the latest changes to the current directory"
473
 
 
474
 
    def get_completer(self, arg, index):
475
 
        if index > 0:
476
 
            return None
477
 
        try:
478
 
            tree = arch.tree_root()
479
 
        except:
480
 
            tree = None
481
 
        return cmdutil.iter_revision_completions(arg, tree)
482
 
    
483
 
    def parse_commandline(self, cmdline, tree):
484
 
        """
485
 
        Parse commandline arguments.  Raises cmdutil.GetHelp if help is needed.
486
 
        
487
 
        :param cmdline: A list of arguments to parse
488
 
        :rtype: (options, Revision, Revision/WorkingTree)
489
 
        """
490
 
        parser=self.get_parser()
491
 
        (options, args) = parser.parse_args(cmdline)
492
 
        if len(args) > 2:
493
 
            raise cmdutil.GetHelp
494
 
 
495
 
        spec=None
496
 
        if len(args)>0:
497
 
            spec=args[0]
498
 
        revision=cmdutil.determine_revision_arch(tree, spec)
499
 
        cmdutil.ensure_archive_registered(revision.archive)
500
 
 
501
 
        mirror_source = cmdutil.get_mirror_source(revision.archive)
502
 
        if mirror_source != None:
503
 
            if cmdutil.prompt("Mirror update"):
504
 
                cmd=cmdutil.mirror_archive(mirror_source, 
505
 
                    revision.archive, arch.NameParser(revision).get_package_version())
506
 
                for line in arch.chatter_classifier(cmd):
507
 
                    cmdutil.colorize(line, options.suppress_chatter)
508
 
 
509
 
                revision=cmdutil.determine_revision_arch(tree, spec)
510
 
 
511
 
        return options, revision 
512
 
 
513
 
    def do_command(self, cmdargs):
514
 
        """
515
 
        Master function that perfoms the "update" command.
516
 
        """
517
 
        tree=arch.tree_root()
518
 
        try:
519
 
            options, to_revision = self.parse_commandline(cmdargs, tree);
520
 
        except cmdutil.CantDetermineRevision, e:
521
 
            print e
522
 
            return
523
 
        except arch.errors.TreeRootError, e:
524
 
            print e
525
 
            return
526
 
        from_revision = arch_compound.tree_latest(tree)
527
 
        if from_revision==to_revision:
528
 
            print "Tree is already up to date with:\n"+str(to_revision)+"."
529
 
            return
530
 
        cmdutil.ensure_archive_registered(from_revision.archive)
531
 
        cmd=cmdutil.apply_delta(from_revision, to_revision, tree,
532
 
            options.patch_forward)
533
 
        for line in cmdutil.iter_apply_delta_filter(cmd):
534
 
            cmdutil.colorize(line)
535
 
        if to_revision.version != tree.tree_version:
536
 
            if cmdutil.prompt("Update version"):
537
 
                tree.tree_version = to_revision.version
538
 
 
539
 
    def get_parser(self):
540
 
        """
541
 
        Returns the options parser to use for the "update" command.
542
 
 
543
 
        :rtype: cmdutil.CmdOptionParser
544
 
        """
545
 
        parser=cmdutil.CmdOptionParser("fai update [options]"
546
 
                                       " [revision/version]")
547
 
        parser.add_option("-f", "--forward", action="store_true", 
548
 
                          dest="patch_forward", default=False, 
549
 
                          help="pass the --forward option to 'patch'")
550
 
        parser.add_option("-s", "--silent", action="store_true", 
551
 
                          dest="suppress_chatter", default=False, 
552
 
                          help="Suppress chatter messages")
553
 
        return parser
554
 
 
555
 
    def help(self, parser=None):
556
 
        """
557
 
        Prints a help message.
558
 
 
559
 
        :param parser: If supplied, the parser to use for generating help.  If \
560
 
        not supplied, it is retrieved.
561
 
        :type parser: cmdutil.CmdOptionParser
562
 
        """
563
 
        if parser is None:
564
 
            parser=self.get_parser()
565
 
        parser.print_help()
566
 
        print """
567
 
Updates a working tree to the current archive revision
568
 
 
569
 
If a revision or version is specified, that is used instead 
570
 
        """
571
 
        help_tree_spec() 
572
 
        return
573
 
 
574
 
 
575
 
class Commit(BaseCommand):
576
 
    """
577
 
    Create a revision based on the changes in the current tree.
578
 
    """
579
 
    
580
 
    def __init__(self):
581
 
        self.description="Write local changes to the archive"
582
 
 
583
 
    def get_completer(self, arg, index):
584
 
        if arg is None:
585
 
            arg = ""
586
 
        return iter_modified_file_completions(arch.tree_root(), arg)
587
 
#        return iter_source_file_completions(arch.tree_root(), arg)
588
 
    
589
 
    def parse_commandline(self, cmdline, tree):
590
 
        """
591
 
        Parse commandline arguments.  Raise cmtutil.GetHelp if help is needed.
592
 
        
593
 
        :param cmdline: A list of arguments to parse
594
 
        :rtype: (options, Revision, Revision/WorkingTree)
595
 
        """
596
 
        parser=self.get_parser()
597
 
        (options, args) = parser.parse_args(cmdline)
598
 
 
599
 
        if len(args) == 0:
600
 
            args = None
601
 
        if options.version is None:
602
 
            return options, tree.tree_version, args
603
 
 
604
 
        revision=cmdutil.determine_revision_arch(tree, options.version)
605
 
        return options, revision.get_version(), args
606
 
 
607
 
    def do_command(self, cmdargs):
608
 
        """
609
 
        Master function that perfoms the "commit" command.
610
 
        """
611
 
        tree=arch.tree_root()
612
 
        options, version, files = self.parse_commandline(cmdargs, tree)
613
 
        ancestor = None
614
 
        if options.__dict__.has_key("base") and options.base:
615
 
            base = cmdutil.determine_revision_tree(tree, options.base)
616
 
            ancestor = base
617
 
        else:
618
 
            base = ancillary.submit_revision(tree)
619
 
            ancestor = base
620
 
        if ancestor is None:
621
 
            ancestor = arch_compound.tree_latest(tree, version)
622
 
 
623
 
        writeversion=version
624
 
        archive=version.archive
625
 
        source=cmdutil.get_mirror_source(archive)
626
 
        allow_old=False
627
 
        writethrough="implicit"
628
 
 
629
 
        if source!=None:
630
 
            if writethrough=="explicit" and \
631
 
                cmdutil.prompt("Writethrough"):
632
 
                writeversion=arch.Version(str(source)+"/"+str(version.get_nonarch()))
633
 
            elif writethrough=="none":
634
 
                raise CommitToMirror(archive)
635
 
 
636
 
        elif archive.is_mirror:
637
 
            raise CommitToMirror(archive)
638
 
 
639
 
        try:
640
 
            last_revision=tree.iter_logs(version, True).next().revision
641
 
        except StopIteration, e:
642
 
            last_revision = None
643
 
            if ancestor is None:
644
 
                if cmdutil.prompt("Import from commit"):
645
 
                    return do_import(version)
646
 
                else:
647
 
                    raise NoVersionLogs(version)
648
 
        try:
649
 
            arch_last_revision = version.iter_revisions(True).next()
650
 
        except StopIteration, e:
651
 
            arch_last_revision = None
652
 
 
653
 
        if last_revision != arch_last_revision:
654
 
            print "Tree is not up to date with %s" % str(version)
655
 
            if not cmdutil.prompt("Out of date"):
656
 
                raise OutOfDate
657
 
            else:
658
 
                allow_old=True
659
 
 
660
 
        try:
661
 
            if not cmdutil.has_changed(ancestor):
662
 
                if not cmdutil.prompt("Empty commit"):
663
 
                    raise EmptyCommit
664
 
        except arch.util.ExecProblem, e:
665
 
            if e.proc.error and e.proc.error.startswith(
666
 
                "missing explicit id for file"):
667
 
                raise MissingID(e)
668
 
            else:
669
 
                raise
670
 
        log = tree.log_message(create=False, version=version)
671
 
        if log is None:
672
 
            try:
673
 
                if cmdutil.prompt("Create log"):
674
 
                    edit_log(tree, version)
675
 
 
676
 
            except cmdutil.NoEditorSpecified, e:
677
 
                raise CommandFailed(e)
678
 
            log = tree.log_message(create=False, version=version)
679
 
        if log is None: 
680
 
            raise NoLogMessage
681
 
        if log["Summary"] is None or len(log["Summary"].strip()) == 0:
682
 
            if not cmdutil.prompt("Omit log summary"):
683
 
                raise errors.NoLogSummary
684
 
        try:
685
 
            for line in tree.iter_commit(version, seal=options.seal_version,
686
 
                base=base, out_of_date_ok=allow_old, file_list=files):
687
 
                cmdutil.colorize(line, options.suppress_chatter)
688
 
 
689
 
        except arch.util.ExecProblem, e:
690
 
            if e.proc.error and e.proc.error.startswith(
691
 
                "These files violate naming conventions:"):
692
 
                raise LintFailure(e.proc.error)
693
 
            else:
694
 
                raise
695
 
 
696
 
    def get_parser(self):
697
 
        """
698
 
        Returns the options parser to use for the "commit" command.
699
 
 
700
 
        :rtype: cmdutil.CmdOptionParser
701
 
        """
702
 
 
703
 
        parser=cmdutil.CmdOptionParser("fai commit [options] [file1]"
704
 
                                       " [file2...]")
705
 
        parser.add_option("--seal", action="store_true", 
706
 
                          dest="seal_version", default=False, 
707
 
                          help="seal this version")
708
 
        parser.add_option("-v", "--version", dest="version", 
709
 
                          help="Use the specified version", 
710
 
                          metavar="VERSION")
711
 
        parser.add_option("-s", "--silent", action="store_true", 
712
 
                          dest="suppress_chatter", default=False, 
713
 
                          help="Suppress chatter messages")
714
 
        if cmdutil.supports_switch("commit", "--base"):
715
 
            parser.add_option("--base", dest="base", help="", 
716
 
                              metavar="REVISION")
717
 
        return parser
718
 
 
719
 
    def help(self, parser=None):
720
 
        """
721
 
        Prints a help message.
722
 
 
723
 
        :param parser: If supplied, the parser to use for generating help.  If \
724
 
        not supplied, it is retrieved.
725
 
        :type parser: cmdutil.CmdOptionParser
726
 
        """
727
 
        if parser is None:
728
 
            parser=self.get_parser()
729
 
        parser.print_help()
730
 
        print """
731
 
Updates a working tree to the current archive revision
732
 
 
733
 
If a version is specified, that is used instead 
734
 
        """
735
 
#        help_tree_spec() 
736
 
        return
737
 
 
738
 
 
739
 
 
740
 
class CatLog(BaseCommand):
741
 
    """
742
 
    Print the log of a given file (from current tree)
743
 
    """
744
 
    def __init__(self):
745
 
        self.description="Prints the patch log for a revision"
746
 
 
747
 
    def get_completer(self, arg, index):
748
 
        if index > 0:
749
 
            return None
750
 
        try:
751
 
            tree = arch.tree_root()
752
 
        except:
753
 
            tree = None
754
 
        return cmdutil.iter_revision_completions(arg, tree)
755
 
 
756
 
    def do_command(self, cmdargs):
757
 
        """
758
 
        Master function that perfoms the "cat-log" command.
759
 
        """
760
 
        parser=self.get_parser()
761
 
        (options, args) = parser.parse_args(cmdargs)
762
 
        try:
763
 
            tree = arch.tree_root()
764
 
        except arch.errors.TreeRootError, e:
765
 
            tree = None
766
 
        spec=None
767
 
        if len(args) > 0:
768
 
            spec=args[0]
769
 
        if len(args) > 1:
770
 
            raise cmdutil.GetHelp()
771
 
        try:
772
 
            if tree:
773
 
                revision = cmdutil.determine_revision_tree(tree, spec)
774
 
            else:
775
 
                revision = cmdutil.determine_revision_arch(tree, spec)
776
 
        except cmdutil.CantDetermineRevision, e:
777
 
            raise CommandFailedWrapper(e)
778
 
        log = None
779
 
        
780
 
        use_tree = (options.source == "tree" or \
781
 
            (options.source == "any" and tree))
782
 
        use_arch = (options.source == "archive" or options.source == "any")
783
 
        
784
 
        log = None
785
 
        if use_tree:
786
 
            for log in tree.iter_logs(revision.get_version()):
787
 
                if log.revision == revision:
788
 
                    break
789
 
                else:
790
 
                    log = None
791
 
        if log is None and use_arch:
792
 
            cmdutil.ensure_revision_exists(revision)
793
 
            log = arch.Patchlog(revision)
794
 
        if log is not None:
795
 
            for item in log.items():
796
 
                print "%s: %s" % item
797
 
            print log.description
798
 
 
799
 
    def get_parser(self):
800
 
        """
801
 
        Returns the options parser to use for the "cat-log" command.
802
 
 
803
 
        :rtype: cmdutil.CmdOptionParser
804
 
        """
805
 
        parser=cmdutil.CmdOptionParser("fai cat-log [revision]")
806
 
        parser.add_option("--archive", action="store_const", dest="source",
807
 
                          const="archive", default="any",
808
 
                          help="Always get the log from the archive")
809
 
        parser.add_option("--tree", action="store_const", dest="source",
810
 
                          const="tree", help="Always get the log from the tree")
811
 
        return parser 
812
 
 
813
 
    def help(self, parser=None):
814
 
        """
815
 
        Prints a help message.
816
 
 
817
 
        :param parser: If supplied, the parser to use for generating help.  If \
818
 
        not supplied, it is retrieved.
819
 
        :type parser: cmdutil.CmdOptionParser
820
 
        """
821
 
        if parser==None:
822
 
            parser=self.get_parser()
823
 
        parser.print_help()
824
 
        print """
825
 
Prints the log for the specified revision
826
 
        """
827
 
        help_tree_spec()
828
 
        return
829
 
 
830
 
class Revert(BaseCommand):
831
 
    """ Reverts a tree (or aspects of it) to a revision
832
 
    """
833
 
    def __init__(self):
834
 
        self.description="Reverts a tree (or aspects of it) to a revision "
835
 
 
836
 
    def get_completer(self, arg, index):
837
 
        if index > 0:
838
 
            return None
839
 
        try:
840
 
            tree = arch.tree_root()
841
 
        except:
842
 
            tree = None
843
 
        return iter_modified_file_completions(tree, arg)
844
 
 
845
 
    def do_command(self, cmdargs):
846
 
        """
847
 
        Master function that perfoms the "revert" command.
848
 
        """
849
 
        parser=self.get_parser()
850
 
        (options, args) = parser.parse_args(cmdargs)
851
 
        try:
852
 
            tree = arch.tree_root()
853
 
        except arch.errors.TreeRootError, e:
854
 
            raise CommandFailed(e)
855
 
        spec=None
856
 
        if options.revision is not None:
857
 
            spec=options.revision
858
 
        try:
859
 
            if spec is not None:
860
 
                revision = cmdutil.determine_revision_tree(tree, spec)
861
 
            else:
862
 
                revision = ancillary.comp_revision(tree)
863
 
        except cmdutil.CantDetermineRevision, e:
864
 
            raise CommandFailedWrapper(e)
865
 
        munger = None
866
 
 
867
 
        if options.file_contents or options.file_perms or options.deletions\
868
 
            or options.additions or options.renames or options.hunk_prompt:
869
 
            munger = arch_compound.MungeOpts()
870
 
            munger.set_hunk_prompt(cmdutil.colorize, cmdutil.user_hunk_confirm,
871
 
                                   options.hunk_prompt)
872
 
 
873
 
        if len(args) > 0 or options.logs or options.pattern_files or \
874
 
            options.control:
875
 
            if munger is None:
876
 
                munger = cmdutil.arch_compound.MungeOpts(True)
877
 
                munger.all_types(True)
878
 
        if len(args) > 0:
879
 
            t_cwd = arch_compound.tree_cwd(tree)
880
 
            for name in args:
881
 
                if len(t_cwd) > 0:
882
 
                    t_cwd += "/"
883
 
                name = "./" + t_cwd + name
884
 
                munger.add_keep_file(name);
885
 
 
886
 
        if options.file_perms:
887
 
            munger.file_perms = True
888
 
        if options.file_contents:
889
 
            munger.file_contents = True
890
 
        if options.deletions:
891
 
            munger.deletions = True
892
 
        if options.additions:
893
 
            munger.additions = True
894
 
        if options.renames:
895
 
            munger.renames = True
896
 
        if options.logs:
897
 
            munger.add_keep_pattern('^\./\{arch\}/[^=].*')
898
 
        if options.control:
899
 
            munger.add_keep_pattern("/\.arch-ids|^\./\{arch\}|"\
900
 
                                    "/\.arch-inventory$")
901
 
        if options.pattern_files:
902
 
            munger.add_keep_pattern(options.pattern_files)
903
 
                
904
 
        for line in arch_compound.revert(tree, revision, munger, 
905
 
                                   not options.no_output):
906
 
            cmdutil.colorize(line)
907
 
 
908
 
 
909
 
    def get_parser(self):
910
 
        """
911
 
        Returns the options parser to use for the "cat-log" command.
912
 
 
913
 
        :rtype: cmdutil.CmdOptionParser
914
 
        """
915
 
        parser=cmdutil.CmdOptionParser("fai revert [options] [FILE...]")
916
 
        parser.add_option("", "--contents", action="store_true", 
917
 
                          dest="file_contents", 
918
 
                          help="Revert file content changes")
919
 
        parser.add_option("", "--permissions", action="store_true", 
920
 
                          dest="file_perms", 
921
 
                          help="Revert file permissions changes")
922
 
        parser.add_option("", "--deletions", action="store_true", 
923
 
                          dest="deletions", 
924
 
                          help="Restore deleted files")
925
 
        parser.add_option("", "--additions", action="store_true", 
926
 
                          dest="additions", 
927
 
                          help="Remove added files")
928
 
        parser.add_option("", "--renames", action="store_true", 
929
 
                          dest="renames", 
930
 
                          help="Revert file names")
931
 
        parser.add_option("--hunks", action="store_true", 
932
 
                          dest="hunk_prompt", default=False,
933
 
                          help="Prompt which hunks to revert")
934
 
        parser.add_option("--pattern-files", dest="pattern_files", 
935
 
                          help="Revert files that match this pattern", 
936
 
                          metavar="REGEX")
937
 
        parser.add_option("--logs", action="store_true", 
938
 
                          dest="logs", default=False,
939
 
                          help="Revert only logs")
940
 
        parser.add_option("--control-files", action="store_true", 
941
 
                          dest="control", default=False,
942
 
                          help="Revert logs and other control files")
943
 
        parser.add_option("-n", "--no-output", action="store_true", 
944
 
                          dest="no_output", 
945
 
                          help="Don't keep an undo changeset")
946
 
        parser.add_option("--revision", dest="revision", 
947
 
                          help="Revert to the specified revision", 
948
 
                          metavar="REVISION")
949
 
        return parser 
950
 
 
951
 
    def help(self, parser=None):
952
 
        """
953
 
        Prints a help message.
954
 
 
955
 
        :param parser: If supplied, the parser to use for generating help.  If \
956
 
        not supplied, it is retrieved.
957
 
        :type parser: cmdutil.CmdOptionParser
958
 
        """
959
 
        if parser==None:
960
 
            parser=self.get_parser()
961
 
        parser.print_help()
962
 
        print """
963
 
Reverts changes in the current working tree.  If no flags are specified, all
964
 
types of changes are reverted.  Otherwise, only selected types of changes are
965
 
reverted.  
966
 
 
967
 
If a revision is specified on the commandline, differences between the current
968
 
tree and that revision are reverted.  If a version is specified, the current
969
 
tree is used to determine the revision.
970
 
 
971
 
If files are specified, only those files listed will have any changes applied.
972
 
To specify a renamed file, you can use either the old or new name. (or both!)
973
 
 
974
 
Unless "-n" is specified, reversions can be undone with "redo".
975
 
        """
976
 
        return
977
 
 
978
 
class Revision(BaseCommand):
979
 
    """
980
 
    Print a revision name based on a revision specifier
981
 
    """
982
 
    def __init__(self):
983
 
        self.description="Prints the name of a revision"
984
 
 
985
 
    def get_completer(self, arg, index):
986
 
        if index > 0:
987
 
            return None
988
 
        try:
989
 
            tree = arch.tree_root()
990
 
        except:
991
 
            tree = None
992
 
        return cmdutil.iter_revision_completions(arg, tree)
993
 
 
994
 
    def do_command(self, cmdargs):
995
 
        """
996
 
        Master function that perfoms the "revision" command.
997
 
        """
998
 
        parser=self.get_parser()
999
 
        (options, args) = parser.parse_args(cmdargs)
1000
 
 
1001
 
        try:
1002
 
            tree = arch.tree_root()
1003
 
        except arch.errors.TreeRootError:
1004
 
            tree = None
1005
 
 
1006
 
        spec=None
1007
 
        if len(args) > 0:
1008
 
            spec=args[0]
1009
 
        if len(args) > 1:
1010
 
            raise cmdutil.GetHelp
1011
 
        try:
1012
 
            if tree:
1013
 
                revision = cmdutil.determine_revision_tree(tree, spec)
1014
 
            else:
1015
 
                revision = cmdutil.determine_revision_arch(tree, spec)
1016
 
        except cmdutil.CantDetermineRevision, e:
1017
 
            print str(e)
1018
 
            return
1019
 
        print options.display(revision)
1020
 
 
1021
 
    def get_parser(self):
1022
 
        """
1023
 
        Returns the options parser to use for the "revision" command.
1024
 
 
1025
 
        :rtype: cmdutil.CmdOptionParser
1026
 
        """
1027
 
        parser=cmdutil.CmdOptionParser("fai revision [revision]")
1028
 
        parser.add_option("", "--location", action="store_const", 
1029
 
                         const=paths.determine_path, dest="display", 
1030
 
                         help="Show location instead of name", default=str)
1031
 
        parser.add_option("--import", action="store_const", 
1032
 
                         const=paths.determine_import_path, dest="display",  
1033
 
                         help="Show location of import file")
1034
 
        parser.add_option("--log", action="store_const", 
1035
 
                         const=paths.determine_log_path, dest="display", 
1036
 
                         help="Show location of log file")
1037
 
        parser.add_option("--patch", action="store_const", 
1038
 
                         dest="display", const=paths.determine_patch_path,
1039
 
                         help="Show location of patchfile")
1040
 
        parser.add_option("--continuation", action="store_const", 
1041
 
                         const=paths.determine_continuation_path, 
1042
 
                         dest="display",
1043
 
                         help="Show location of continuation file")
1044
 
        parser.add_option("--cacherev", action="store_const", 
1045
 
                         const=paths.determine_cacherev_path, dest="display",
1046
 
                         help="Show location of cacherev file")
1047
 
        return parser 
1048
 
 
1049
 
    def help(self, parser=None):
1050
 
        """
1051
 
        Prints a help message.
1052
 
 
1053
 
        :param parser: If supplied, the parser to use for generating help.  If \
1054
 
        not supplied, it is retrieved.
1055
 
        :type parser: cmdutil.CmdOptionParser
1056
 
        """
1057
 
        if parser==None:
1058
 
            parser=self.get_parser()
1059
 
        parser.print_help()
1060
 
        print """
1061
 
Expands aliases and prints the name of the specified revision.  Instead of
1062
 
the name, several options can be used to print locations.  If more than one is
1063
 
specified, the last one is used.
1064
 
        """
1065
 
        help_tree_spec()
1066
 
        return
1067
 
 
1068
 
class Revisions(BaseCommand):
1069
 
    """
1070
 
    Print a revision name based on a revision specifier
1071
 
    """
1072
 
    def __init__(self):
1073
 
        self.description="Lists revisions"
1074
 
        self.cl_revisions = []
1075
 
    
1076
 
    def do_command(self, cmdargs):
1077
 
        """
1078
 
        Master function that perfoms the "revision" command.
1079
 
        """
1080
 
        (options, args) = self.get_parser().parse_args(cmdargs)
1081
 
        if len(args) > 1:
1082
 
            raise cmdutil.GetHelp
1083
 
        try:
1084
 
            self.tree = arch.tree_root()
1085
 
        except arch.errors.TreeRootError:
1086
 
            self.tree = None
1087
 
        if options.type == "default":
1088
 
            options.type = "archive"
1089
 
        try:
1090
 
            iter = cmdutil.revision_iterator(self.tree, options.type, args, 
1091
 
                                             options.reverse, options.modified,
1092
 
                                             options.shallow)
1093
 
        except cmdutil.CantDetermineRevision, e:
1094
 
            raise CommandFailedWrapper(e)
1095
 
        except cmdutil.CantDetermineVersion, e:
1096
 
            raise CommandFailedWrapper(e)
1097
 
        if options.skip is not None:
1098
 
            iter = cmdutil.iter_skip(iter, int(options.skip))
1099
 
 
1100
 
        try:
1101
 
            for revision in iter:
1102
 
                log = None
1103
 
                if isinstance(revision, arch.Patchlog):
1104
 
                    log = revision
1105
 
                    revision=revision.revision
1106
 
                out = options.display(revision)
1107
 
                if out is not None:
1108
 
                    print out
1109
 
                if log is None and (options.summary or options.creator or 
1110
 
                                    options.date or options.merges):
1111
 
                    log = revision.patchlog
1112
 
                if options.creator:
1113
 
                    print "    %s" % log.creator
1114
 
                if options.date:
1115
 
                    print "    %s" % time.strftime('%Y-%m-%d %H:%M:%S %Z', log.date)
1116
 
                if options.summary:
1117
 
                    print "    %s" % log.summary
1118
 
                if options.merges:
1119
 
                    showed_title = False
1120
 
                    for revision in log.merged_patches:
1121
 
                        if not showed_title:
1122
 
                            print "    Merged:"
1123
 
                            showed_title = True
1124
 
                        print "    %s" % revision
1125
 
            if len(self.cl_revisions) > 0:
1126
 
                print pylon.changelog_for_merge(self.cl_revisions)
1127
 
        except pylon.errors.TreeRootNone:
1128
 
            raise CommandFailedWrapper(
1129
 
                Exception("This option can only be used in a project tree."))
1130
 
 
1131
 
    def changelog_append(self, revision):
1132
 
        if isinstance(revision, arch.Revision):
1133
 
            revision=arch.Patchlog(revision)
1134
 
        self.cl_revisions.append(revision)
1135
 
   
1136
 
    def get_parser(self):
1137
 
        """
1138
 
        Returns the options parser to use for the "revision" command.
1139
 
 
1140
 
        :rtype: cmdutil.CmdOptionParser
1141
 
        """
1142
 
        parser=cmdutil.CmdOptionParser("fai revisions [version/revision]")
1143
 
        select = cmdutil.OptionGroup(parser, "Selection options",
1144
 
                          "Control which revisions are listed.  These options"
1145
 
                          " are mutually exclusive.  If more than one is"
1146
 
                          " specified, the last is used.")
1147
 
 
1148
 
        cmdutil.add_revision_iter_options(select)
1149
 
        parser.add_option("", "--skip", dest="skip", 
1150
 
                          help="Skip revisions.  Positive numbers skip from "
1151
 
                          "beginning, negative skip from end.",
1152
 
                          metavar="NUMBER")
1153
 
 
1154
 
        parser.add_option_group(select)
1155
 
 
1156
 
        format = cmdutil.OptionGroup(parser, "Revision format options",
1157
 
                          "These control the appearance of listed revisions")
1158
 
        format.add_option("", "--location", action="store_const", 
1159
 
                         const=paths.determine_path, dest="display", 
1160
 
                         help="Show location instead of name", default=str)
1161
 
        format.add_option("--import", action="store_const", 
1162
 
                         const=paths.determine_import_path, dest="display",  
1163
 
                         help="Show location of import file")
1164
 
        format.add_option("--log", action="store_const", 
1165
 
                         const=paths.determine_log_path, dest="display", 
1166
 
                         help="Show location of log file")
1167
 
        format.add_option("--patch", action="store_const", 
1168
 
                         dest="display", const=paths.determine_patch_path,
1169
 
                         help="Show location of patchfile")
1170
 
        format.add_option("--continuation", action="store_const", 
1171
 
                         const=paths.determine_continuation_path, 
1172
 
                         dest="display",
1173
 
                         help="Show location of continuation file")
1174
 
        format.add_option("--cacherev", action="store_const", 
1175
 
                         const=paths.determine_cacherev_path, dest="display",
1176
 
                         help="Show location of cacherev file")
1177
 
        format.add_option("--changelog", action="store_const", 
1178
 
                         const=self.changelog_append, dest="display",
1179
 
                         help="Show location of cacherev file")
1180
 
        parser.add_option_group(format)
1181
 
        display = cmdutil.OptionGroup(parser, "Display format options",
1182
 
                          "These control the display of data")
1183
 
        display.add_option("-r", "--reverse", action="store_true", 
1184
 
                          dest="reverse", help="Sort from newest to oldest")
1185
 
        display.add_option("-s", "--summary", action="store_true", 
1186
 
                          dest="summary", help="Show patchlog summary")
1187
 
        display.add_option("-D", "--date", action="store_true", 
1188
 
                          dest="date", help="Show patchlog date")
1189
 
        display.add_option("-c", "--creator", action="store_true", 
1190
 
                          dest="creator", help="Show the id that committed the"
1191
 
                          " revision")
1192
 
        display.add_option("-m", "--merges", action="store_true", 
1193
 
                          dest="merges", help="Show the revisions that were"
1194
 
                          " merged")
1195
 
        parser.add_option_group(display)
1196
 
        return parser 
1197
 
    def help(self, parser=None):
1198
 
        """Attempt to explain the revisions command
1199
 
        
1200
 
        :param parser: If supplied, used to determine options
1201
 
        """
1202
 
        if parser==None:
1203
 
            parser=self.get_parser()
1204
 
        parser.print_help()
1205
 
        print """List revisions.
1206
 
        """
1207
 
        help_tree_spec()
1208
 
 
1209
 
 
1210
 
class Get(BaseCommand):
1211
 
    """
1212
 
    Retrieve a revision from the archive
1213
 
    """
1214
 
    def __init__(self):
1215
 
        self.description="Retrieve a revision from the archive"
1216
 
        self.parser=self.get_parser()
1217
 
 
1218
 
 
1219
 
    def get_completer(self, arg, index):
1220
 
        if index > 0:
1221
 
            return None
1222
 
        try:
1223
 
            tree = arch.tree_root()
1224
 
        except:
1225
 
            tree = None
1226
 
        return cmdutil.iter_revision_completions(arg, tree)
1227
 
 
1228
 
 
1229
 
    def do_command(self, cmdargs):
1230
 
        """
1231
 
        Master function that perfoms the "get" command.
1232
 
        """
1233
 
        (options, args) = self.parser.parse_args(cmdargs)
1234
 
        if len(args) < 1:
1235
 
            return self.help()            
1236
 
        try:
1237
 
            tree = arch.tree_root()
1238
 
        except arch.errors.TreeRootError:
1239
 
            tree = None
1240
 
        
1241
 
        arch_loc = None
1242
 
        try:
1243
 
            revision, arch_loc = paths.full_path_decode(args[0])
1244
 
        except Exception, e:
1245
 
            revision = cmdutil.determine_revision_arch(tree, args[0], 
1246
 
                check_existence=False, allow_package=True)
1247
 
        if len(args) > 1:
1248
 
            directory = args[1]
1249
 
        else:
1250
 
            directory = str(revision.nonarch)
1251
 
        if os.path.exists(directory):
1252
 
            raise DirectoryExists(directory)
1253
 
        cmdutil.ensure_archive_registered(revision.archive, arch_loc)
1254
 
        try:
1255
 
            cmdutil.ensure_revision_exists(revision)
1256
 
        except cmdutil.NoSuchRevision, e:
1257
 
            raise CommandFailedWrapper(e)
1258
 
 
1259
 
        link = cmdutil.prompt ("get link")
1260
 
        for line in cmdutil.iter_get(revision, directory, link,
1261
 
                                     options.no_pristine,
1262
 
                                     options.no_greedy_add):
1263
 
            cmdutil.colorize(line)
1264
 
 
1265
 
    def get_parser(self):
1266
 
        """
1267
 
        Returns the options parser to use for the "get" command.
1268
 
 
1269
 
        :rtype: cmdutil.CmdOptionParser
1270
 
        """
1271
 
        parser=cmdutil.CmdOptionParser("fai get revision [dir]")
1272
 
        parser.add_option("--no-pristine", action="store_true", 
1273
 
                         dest="no_pristine", 
1274
 
                         help="Do not make pristine copy for reference")
1275
 
        parser.add_option("--no-greedy-add", action="store_true", 
1276
 
                         dest="no_greedy_add", 
1277
 
                         help="Never add to greedy libraries")
1278
 
 
1279
 
        return parser 
1280
 
 
1281
 
    def help(self, parser=None):
1282
 
        """
1283
 
        Prints a help message.
1284
 
 
1285
 
        :param parser: If supplied, the parser to use for generating help.  If \
1286
 
        not supplied, it is retrieved.
1287
 
        :type parser: cmdutil.CmdOptionParser
1288
 
        """
1289
 
        if parser==None:
1290
 
            parser=self.get_parser()
1291
 
        parser.print_help()
1292
 
        print """
1293
 
Expands aliases and constructs a project tree for a revision.  If the optional
1294
 
"dir" argument is provided, the project tree will be stored in this directory.
1295
 
        """
1296
 
        help_tree_spec()
1297
 
        return
1298
 
 
1299
 
class PromptCmd(cmd.Cmd):
1300
 
    def __init__(self):
1301
 
        cmd.Cmd.__init__(self)
1302
 
        self.prompt = "Fai> "
1303
 
        try:
1304
 
            self.tree = arch.tree_root()
1305
 
        except:
1306
 
            self.tree = None
1307
 
        self.set_title()
1308
 
        self.set_prompt()
1309
 
        self.fake_aba = abacmds.AbaCmds()
1310
 
        self.identchars += '-'
1311
 
        self.history_file = os.path.expanduser("~/.fai-history")
1312
 
        readline.set_completer_delims(string.whitespace)
1313
 
        if os.access(self.history_file, os.R_OK) and \
1314
 
            os.path.isfile(self.history_file):
1315
 
            readline.read_history_file(self.history_file)
1316
 
        self.cwd = os.getcwd()
1317
 
 
1318
 
    def write_history(self):
1319
 
        readline.write_history_file(self.history_file)
1320
 
 
1321
 
    def do_quit(self, args):
1322
 
        self.write_history()
1323
 
        sys.exit(0)
1324
 
 
1325
 
    def do_exit(self, args):
1326
 
        self.do_quit(args)
1327
 
 
1328
 
    def do_EOF(self, args):
1329
 
        print
1330
 
        self.do_quit(args)
1331
 
 
1332
 
    def postcmd(self, line, bar):
1333
 
        self.set_title()
1334
 
        self.set_prompt()
1335
 
 
1336
 
    def set_prompt(self):
1337
 
        if self.tree is not None:
1338
 
            try:
1339
 
                prompt = pylon.alias_or_version(self.tree.tree_version, 
1340
 
                                                self.tree, 
1341
 
                                                full=False)
1342
 
                if prompt is not None:
1343
 
                    prompt = " " + prompt
1344
 
            except:
1345
 
                prompt = ""
1346
 
        else:
1347
 
            prompt = ""
1348
 
        self.prompt = "Fai%s> " % prompt
1349
 
 
1350
 
    def set_title(self, command=None):
1351
 
        try:
1352
 
            version = pylon.alias_or_version(self.tree.tree_version, self.tree, 
1353
 
                                             full=False)
1354
 
        except:
1355
 
            version = "[no version]"
1356
 
        if command is None:
1357
 
            command = ""
1358
 
        sys.stdout.write(terminal.term_title("Fai %s %s" % (command, version)))
1359
 
 
1360
 
    def do_cd(self, line):
1361
 
        if line == "":
1362
 
            line = "~"
1363
 
        line = os.path.expanduser(line)
1364
 
        if os.path.isabs(line):
1365
 
            newcwd = line
1366
 
        else:
1367
 
            newcwd = self.cwd+'/'+line
1368
 
        newcwd = os.path.normpath(newcwd)
1369
 
        try:
1370
 
            os.chdir(newcwd)
1371
 
            self.cwd = newcwd
1372
 
        except Exception, e:
1373
 
            print e
1374
 
        try:
1375
 
            self.tree = arch.tree_root()
1376
 
        except:
1377
 
            self.tree = None
1378
 
 
1379
 
    def do_help(self, line):
1380
 
        Help()(line)
1381
 
 
1382
 
    def default(self, line):
1383
 
        args = line.split()
1384
 
        if find_command(args[0]):
1385
 
            try:
1386
 
                find_command(args[0]).do_command(args[1:])
1387
 
            except cmdutil.BadCommandOption, e:
1388
 
                print e
1389
 
            except cmdutil.GetHelp, e:
1390
 
                find_command(args[0]).help()
1391
 
            except CommandFailed, e:
1392
 
                print e
1393
 
            except arch.errors.ArchiveNotRegistered, e:
1394
 
                print e
1395
 
            except KeyboardInterrupt, e:
1396
 
                print "Interrupted"
1397
 
            except arch.util.ExecProblem, e:
1398
 
                print e.proc.error.rstrip('\n')
1399
 
            except cmdutil.CantDetermineVersion, e:
1400
 
                print e
1401
 
            except cmdutil.CantDetermineRevision, e:
1402
 
                print e
1403
 
            except Exception, e:
1404
 
                print "Unhandled error:\n%s" % errors.exception_str(e)
1405
 
 
1406
 
        elif suggestions.has_key(args[0]):
1407
 
            print suggestions[args[0]]
1408
 
 
1409
 
        elif self.fake_aba.is_command(args[0]):
1410
 
            tree = None
1411
 
            try:
1412
 
                tree = arch.tree_root()
1413
 
            except arch.errors.TreeRootError:
1414
 
                pass
1415
 
            cmd = self.fake_aba.is_command(args[0])
1416
 
            try:
1417
 
                cmd.run(cmdutil.expand_prefix_alias(args[1:], tree))
1418
 
            except KeyboardInterrupt, e:
1419
 
                print "Interrupted"
1420
 
 
1421
 
        elif options.tla_fallthrough and args[0] != "rm" and \
1422
 
            cmdutil.is_tla_command(args[0]):
1423
 
            try:
1424
 
                tree = None
1425
 
                try:
1426
 
                    tree = arch.tree_root()
1427
 
                except arch.errors.TreeRootError:
1428
 
                    pass
1429
 
                args = cmdutil.expand_prefix_alias(args, tree)
1430
 
                arch.util.exec_safe('tla', args, stderr=sys.stderr,
1431
 
                expected=(0, 1))
1432
 
            except arch.util.ExecProblem, e:
1433
 
                pass
1434
 
            except KeyboardInterrupt, e:
1435
 
                print "Interrupted"
1436
 
        else:
1437
 
            try:
1438
 
                try:
1439
 
                    tree = arch.tree_root()
1440
 
                except arch.errors.TreeRootError:
1441
 
                    tree = None
1442
 
                args=line.split()
1443
 
                os.system(" ".join(cmdutil.expand_prefix_alias(args, tree)))
1444
 
            except KeyboardInterrupt, e:
1445
 
                print "Interrupted"
1446
 
 
1447
 
    def completenames(self, text, line, begidx, endidx):
1448
 
        completions = []
1449
 
        iter = iter_command_names(self.fake_aba)
1450
 
        try:
1451
 
            if len(line) > 0:
1452
 
                arg = line.split()[-1]
1453
 
            else:
1454
 
                arg = ""
1455
 
            iter = cmdutil.iter_munged_completions(iter, arg, text)
1456
 
        except Exception, e:
1457
 
            print e
1458
 
        return list(iter)
1459
 
 
1460
 
    def completedefault(self, text, line, begidx, endidx):
1461
 
        """Perform completion for native commands.
1462
 
        
1463
 
        :param text: The text to complete
1464
 
        :type text: str
1465
 
        :param line: The entire line to complete
1466
 
        :type line: str
1467
 
        :param begidx: The start of the text in the line
1468
 
        :type begidx: int
1469
 
        :param endidx: The end of the text in the line
1470
 
        :type endidx: int
1471
 
        """
1472
 
        try:
1473
 
            (cmd, args, foo) = self.parseline(line)
1474
 
            command_obj=find_command(cmd)
1475
 
            if command_obj is not None:
1476
 
                return command_obj.complete(args.split(), text)
1477
 
            elif not self.fake_aba.is_command(cmd) and \
1478
 
                cmdutil.is_tla_command(cmd):
1479
 
                iter = cmdutil.iter_supported_switches(cmd)
1480
 
                if len(args) > 0:
1481
 
                    arg = args.split()[-1]
1482
 
                else:
1483
 
                    arg = ""
1484
 
                if arg.startswith("-"):
1485
 
                    return list(cmdutil.iter_munged_completions(iter, arg, 
1486
 
                                                                text))
1487
 
                else:
1488
 
                    return list(cmdutil.iter_munged_completions(
1489
 
                        cmdutil.iter_file_completions(arg), arg, text))
1490
 
 
1491
 
 
1492
 
            elif cmd == "cd":
1493
 
                if len(args) > 0:
1494
 
                    arg = args.split()[-1]
1495
 
                else:
1496
 
                    arg = ""
1497
 
                iter = cmdutil.iter_dir_completions(arg)
1498
 
                iter = cmdutil.iter_munged_completions(iter, arg, text)
1499
 
                return list(iter)
1500
 
            elif len(args)>0:
1501
 
                arg = args.split()[-1]
1502
 
                iter = cmdutil.iter_file_completions(arg)
1503
 
                return list(cmdutil.iter_munged_completions(iter, arg, text))
1504
 
            else:
1505
 
                return self.completenames(text, line, begidx, endidx)
1506
 
        except Exception, e:
1507
 
            print e
1508
 
 
1509
 
 
1510
 
def iter_command_names(fake_aba):
1511
 
    for entry in cmdutil.iter_combine([commands.iterkeys(), 
1512
 
                                     fake_aba.get_commands(), 
1513
 
                                     cmdutil.iter_tla_commands(False)]):
1514
 
        if not suggestions.has_key(str(entry)):
1515
 
            yield entry
1516
 
 
1517
 
 
1518
 
def iter_source_file_completions(tree, arg):
1519
 
    treepath = arch_compound.tree_cwd(tree)
1520
 
    if len(treepath) > 0:
1521
 
        dirs = [treepath]
1522
 
    else:
1523
 
        dirs = None
1524
 
    for file in tree.iter_inventory(dirs, source=True, both=True):
1525
 
        file = file_completion_match(file, treepath, arg)
1526
 
        if file is not None:
1527
 
            yield file
1528
 
 
1529
 
 
1530
 
def iter_untagged(tree, dirs):
1531
 
    for file in arch_core.iter_inventory_filter(tree, dirs, tagged=False, 
1532
 
                                                categories=arch_core.non_root,
1533
 
                                                control_files=True):
1534
 
        yield file.name 
1535
 
 
1536
 
 
1537
 
def iter_untagged_completions(tree, arg):
1538
 
    """Generate an iterator for all visible untagged files that match arg.
1539
 
 
1540
 
    :param tree: The tree to look for untagged files in
1541
 
    :type tree: `arch.WorkingTree`
1542
 
    :param arg: The argument to match
1543
 
    :type arg: str
1544
 
    :return: An iterator of all matching untagged files
1545
 
    :rtype: iterator of str
1546
 
    """
1547
 
    treepath = arch_compound.tree_cwd(tree)
1548
 
    if len(treepath) > 0:
1549
 
        dirs = [treepath]
1550
 
    else:
1551
 
        dirs = None
1552
 
 
1553
 
    for file in iter_untagged(tree, dirs):
1554
 
        file = file_completion_match(file, treepath, arg)
1555
 
        if file is not None:
1556
 
            yield file
1557
 
 
1558
 
 
1559
 
def file_completion_match(file, treepath, arg):
1560
 
    """Determines whether a file within an arch tree matches the argument.
1561
 
 
1562
 
    :param file: The rooted filename
1563
 
    :type file: str
1564
 
    :param treepath: The path to the cwd within the tree
1565
 
    :type treepath: str
1566
 
    :param arg: The prefix to match
1567
 
    :return: The completion name, or None if not a match
1568
 
    :rtype: str
1569
 
    """
1570
 
    if not file.startswith(treepath):
1571
 
        return None
1572
 
    if treepath != "":
1573
 
        file = file[len(treepath)+1:]
1574
 
 
1575
 
    if not file.startswith(arg):
1576
 
        return None 
1577
 
    if os.path.isdir(file):
1578
 
        file += '/'
1579
 
    return file
1580
 
 
1581
 
def iter_modified_file_completions(tree, arg):
1582
 
    """Returns a list of modified files that match the specified prefix.
1583
 
 
1584
 
    :param tree: The current tree
1585
 
    :type tree: `arch.WorkingTree`
1586
 
    :param arg: The prefix to match
1587
 
    :type arg: str
1588
 
    """
1589
 
    treepath = arch_compound.tree_cwd(tree)
1590
 
    tmpdir = util.tmpdir()
1591
 
    changeset = tmpdir+"/changeset"
1592
 
    completions = []
1593
 
    revision = cmdutil.determine_revision_tree(tree)
1594
 
    for line in arch.iter_delta(revision, tree, changeset):
1595
 
        if isinstance(line, arch.FileModification):
1596
 
            file = file_completion_match(line.name[1:], treepath, arg)
1597
 
            if file is not None:
1598
 
                completions.append(file)
1599
 
    shutil.rmtree(tmpdir)
1600
 
    return completions
1601
 
 
1602
 
class Shell(BaseCommand):
1603
 
    def __init__(self):
1604
 
        self.description = "Runs Fai as a shell"
1605
 
 
1606
 
    def do_command(self, cmdargs):
1607
 
        if len(cmdargs)!=0:
1608
 
            raise cmdutil.GetHelp
1609
 
        prompt = PromptCmd()
1610
 
        try:
1611
 
            prompt.cmdloop()
1612
 
        finally:
1613
 
            prompt.write_history()
1614
 
 
1615
 
class AddID(BaseCommand):
1616
 
    """
1617
 
    Adds an inventory id for the given file
1618
 
    """
1619
 
    def __init__(self):
1620
 
        self.description="Add an inventory id for a given file"
1621
 
 
1622
 
    def get_completer(self, arg, index):
1623
 
        tree = arch.tree_root()
1624
 
        return iter_untagged_completions(tree, arg)
1625
 
 
1626
 
    def do_command(self, cmdargs):
1627
 
        """
1628
 
        Master function that perfoms the "revision" command.
1629
 
        """
1630
 
        parser=self.get_parser()
1631
 
        (options, args) = parser.parse_args(cmdargs)
1632
 
 
1633
 
        try:
1634
 
            tree = arch.tree_root()
1635
 
        except arch.errors.TreeRootError, e:
1636
 
            raise pylon.errors.CommandFailedWrapper(e)
1637
 
            
1638
 
 
1639
 
        if (len(args) == 0) == (options.untagged == False):
1640
 
            raise cmdutil.GetHelp
1641
 
 
1642
 
       #if options.id and len(args) != 1:
1643
 
       #    print "If --id is specified, only one file can be named."
1644
 
       #    return
1645
 
        
1646
 
        method = tree.tagging_method
1647
 
        
1648
 
        if options.id_type == "tagline":
1649
 
            if method != "tagline":
1650
 
                if not cmdutil.prompt("Tagline in other tree"):
1651
 
                    if method == "explicit" or method == "implicit":
1652
 
                        options.id_type == method
1653
 
                    else:
1654
 
                        print "add-id not supported for \"%s\" tagging method"\
1655
 
                            % method 
1656
 
                        return
1657
 
        
1658
 
        elif options.id_type == "implicit":
1659
 
            if method != "implicit":
1660
 
                if not cmdutil.prompt("Implicit in other tree"):
1661
 
                    if method == "explicit" or method == "tagline":
1662
 
                        options.id_type == method
1663
 
                    else:
1664
 
                        print "add-id not supported for \"%s\" tagging method"\
1665
 
                            % method 
1666
 
                        return
1667
 
        elif options.id_type == "explicit":
1668
 
            if method != "tagline" and method != explicit:
1669
 
                if not prompt("Explicit in other tree"):
1670
 
                    print "add-id not supported for \"%s\" tagging method" % \
1671
 
                        method
1672
 
                    return
1673
 
        
1674
 
        if options.id_type == "auto":
1675
 
            if method != "tagline" and method != "explicit" \
1676
 
                and method !="implicit":
1677
 
                print "add-id not supported for \"%s\" tagging method" % method
1678
 
                return
1679
 
            else:
1680
 
                options.id_type = method
1681
 
        if options.untagged:
1682
 
            args = None
1683
 
        self.add_ids(tree, options.id_type, args)
1684
 
 
1685
 
    def add_ids(self, tree, id_type, files=()):
1686
 
        """Add inventory ids to files.
1687
 
        
1688
 
        :param tree: the tree the files are in
1689
 
        :type tree: `arch.WorkingTree`
1690
 
        :param id_type: the type of id to add: "explicit" or "tagline"
1691
 
        :type id_type: str
1692
 
        :param files: The list of files to add.  If None do all untagged.
1693
 
        :type files: tuple of str
1694
 
        """
1695
 
 
1696
 
        untagged = (files is None)
1697
 
        if untagged:
1698
 
            files = list(iter_untagged(tree, None))
1699
 
        previous_files = []
1700
 
        while len(files) > 0:
1701
 
            previous_files.extend(files)
1702
 
            if id_type == "explicit":
1703
 
                cmdutil.add_id(files)
1704
 
            elif id_type == "tagline" or id_type == "implicit":
1705
 
                for file in files:
1706
 
                    try:
1707
 
                        implicit = (id_type == "implicit")
1708
 
                        cmdutil.add_tagline_or_explicit_id(file, False,
1709
 
                                                           implicit)
1710
 
                    except cmdutil.AlreadyTagged:
1711
 
                        print "\"%s\" already has a tagline." % file
1712
 
                    except cmdutil.NoCommentSyntax:
1713
 
                        pass
1714
 
            #do inventory after tagging until no untagged files are encountered
1715
 
            if untagged:
1716
 
                files = []
1717
 
                for file in iter_untagged(tree, None):
1718
 
                    if not file in previous_files:
1719
 
                        files.append(file)
1720
 
 
1721
 
            else:
1722
 
                break
1723
 
 
1724
 
    def get_parser(self):
1725
 
        """
1726
 
        Returns the options parser to use for the "revision" command.
1727
 
 
1728
 
        :rtype: cmdutil.CmdOptionParser
1729
 
        """
1730
 
        parser=cmdutil.CmdOptionParser("fai add-id file1 [file2] [file3]...")
1731
 
# ddaa suggests removing this to promote GUIDs.  Let's see who squalks.
1732
 
#        parser.add_option("-i", "--id", dest="id", 
1733
 
#                         help="Specify id for a single file", default=None)
1734
 
        parser.add_option("--tltl", action="store_true", 
1735
 
                         dest="lord_style",  help="Use Tom Lord's style of id.")
1736
 
        parser.add_option("--explicit", action="store_const", 
1737
 
                         const="explicit", dest="id_type", 
1738
 
                         help="Use an explicit id", default="auto")
1739
 
        parser.add_option("--tagline", action="store_const", 
1740
 
                         const="tagline", dest="id_type", 
1741
 
                         help="Use a tagline id")
1742
 
        parser.add_option("--implicit", action="store_const", 
1743
 
                         const="implicit", dest="id_type", 
1744
 
                         help="Use an implicit id (deprecated)")
1745
 
        parser.add_option("--untagged", action="store_true", 
1746
 
                         dest="untagged", default=False, 
1747
 
                         help="tag all untagged files")
1748
 
        return parser 
1749
 
 
1750
 
    def help(self, parser=None):
1751
 
        """
1752
 
        Prints a help message.
1753
 
 
1754
 
        :param parser: If supplied, the parser to use for generating help.  If \
1755
 
        not supplied, it is retrieved.
1756
 
        :type parser: cmdutil.CmdOptionParser
1757
 
        """
1758
 
        if parser==None:
1759
 
            parser=self.get_parser()
1760
 
        parser.print_help()
1761
 
        print """
1762
 
Adds an inventory to the specified file(s) and directories.  If --untagged is
1763
 
specified, adds inventory to all untagged files and directories.
1764
 
        """
1765
 
        return
1766
 
 
1767
 
 
1768
 
class Merge(BaseCommand):
1769
 
    """
1770
 
    Merges changes from other versions into the current tree
1771
 
    """
1772
 
    def __init__(self):
1773
 
        self.description="Merges changes from other versions"
1774
 
        try:
1775
 
            self.tree = arch.tree_root()
1776
 
        except:
1777
 
            self.tree = None
1778
 
 
1779
 
 
1780
 
    def get_completer(self, arg, index):
1781
 
        if self.tree is None:
1782
 
            raise arch.errors.TreeRootError
1783
 
        return cmdutil.merge_completions(self.tree, arg, index)
1784
 
 
1785
 
    def do_command(self, cmdargs):
1786
 
        """
1787
 
        Master function that perfoms the "merge" command.
1788
 
        """
1789
 
        parser=self.get_parser()
1790
 
        (options, args) = parser.parse_args(cmdargs)
1791
 
        if options.diff3:
1792
 
            action="star-merge"
1793
 
        else:
1794
 
            action = options.action
1795
 
        
1796
 
        if self.tree is None:
1797
 
            raise arch.errors.TreeRootError(os.getcwd())
1798
 
        if cmdutil.has_changed(ancillary.comp_revision(self.tree)):
1799
 
            raise UncommittedChanges(self.tree)
1800
 
 
1801
 
        if len(args) > 0:
1802
 
            revisions = []
1803
 
            for arg in args:
1804
 
                revisions.append(cmdutil.determine_revision_arch(self.tree, 
1805
 
                                                                 arg))
1806
 
            source = "from commandline"
1807
 
        else:
1808
 
            revisions = ancillary.iter_partner_revisions(self.tree, 
1809
 
                                                         self.tree.tree_version)
1810
 
            source = "from partner version"
1811
 
        revisions = misc.rewind_iterator(revisions)
1812
 
        try:
1813
 
            revisions.next()
1814
 
            revisions.rewind()
1815
 
        except StopIteration, e:
1816
 
            revision = cmdutil.tag_cur(self.tree)
1817
 
            if revision is None:
1818
 
                raise CantDetermineRevision("", "No version specified, no "
1819
 
                                            "partner-versions, and no tag"
1820
 
                                            " source")
1821
 
            revisions = [revision]
1822
 
            source = "from tag source"
1823
 
        for revision in revisions:
1824
 
            cmdutil.ensure_archive_registered(revision.archive)
1825
 
            cmdutil.colorize(arch.Chatter("* Merging %s [%s]" % 
1826
 
                             (revision, source)))
1827
 
            if action=="native-merge" or action=="update":
1828
 
                if self.native_merge(revision, action) == 0:
1829
 
                    continue
1830
 
            elif action=="star-merge":
1831
 
                try: 
1832
 
                    self.star_merge(revision, options.diff3)
1833
 
                except errors.MergeProblem, e:
1834
 
                    break
1835
 
            if cmdutil.has_changed(self.tree.tree_version):
1836
 
                break
1837
 
 
1838
 
    def star_merge(self, revision, diff3):
1839
 
        """Perform a star-merge on the current tree.
1840
 
        
1841
 
        :param revision: The revision to use for the merge
1842
 
        :type revision: `arch.Revision`
1843
 
        :param diff3: If true, do a diff3 merge
1844
 
        :type diff3: bool
1845
 
        """
1846
 
        try:
1847
 
            for line in self.tree.iter_star_merge(revision, diff3=diff3):
1848
 
                cmdutil.colorize(line)
1849
 
        except arch.util.ExecProblem, e:
1850
 
            if e.proc.status is not None and e.proc.status == 1:
1851
 
                if e.proc.error:
1852
 
                    print e.proc.error
1853
 
                raise MergeProblem
1854
 
            else:
1855
 
                raise
1856
 
 
1857
 
    def native_merge(self, other_revision, action):
1858
 
        """Perform a native-merge on the current tree.
1859
 
        
1860
 
        :param other_revision: The revision to use for the merge
1861
 
        :type other_revision: `arch.Revision`
1862
 
        :return: 0 if the merge was skipped, 1 if it was applied
1863
 
        """
1864
 
        other_tree = arch_compound.find_or_make_local_revision(other_revision)
1865
 
        try:
1866
 
            if action == "native-merge":
1867
 
                ancestor = arch_compound.merge_ancestor2(self.tree, other_tree, 
1868
 
                                                         other_revision)
1869
 
            elif action == "update":
1870
 
                ancestor = arch_compound.tree_latest(self.tree, 
1871
 
                                                     other_revision.version)
1872
 
        except CantDetermineRevision, e:
1873
 
            raise CommandFailedWrapper(e)
1874
 
        cmdutil.colorize(arch.Chatter("* Found common ancestor %s" % ancestor))
1875
 
        if (ancestor == other_revision):
1876
 
            cmdutil.colorize(arch.Chatter("* Skipping redundant merge" 
1877
 
                                          % ancestor))
1878
 
            return 0
1879
 
        delta = cmdutil.apply_delta(ancestor, other_tree, self.tree)    
1880
 
        for line in cmdutil.iter_apply_delta_filter(delta):
1881
 
            cmdutil.colorize(line)
1882
 
        return 1
1883
 
 
1884
 
 
1885
 
 
1886
 
    def get_parser(self):
1887
 
        """
1888
 
        Returns the options parser to use for the "merge" command.
1889
 
 
1890
 
        :rtype: cmdutil.CmdOptionParser
1891
 
        """
1892
 
        parser=cmdutil.CmdOptionParser("fai merge [VERSION]")
1893
 
        parser.add_option("-s", "--star-merge", action="store_const",
1894
 
                          dest="action", help="Use star-merge",
1895
 
                          const="star-merge", default="native-merge")
1896
 
        parser.add_option("--update", action="store_const",
1897
 
                          dest="action", help="Use update picker",
1898
 
                          const="update")
1899
 
        parser.add_option("--diff3", action="store_true", 
1900
 
                         dest="diff3",  
1901
 
                         help="Use diff3 for merge (implies star-merge)")
1902
 
        return parser 
1903
 
 
1904
 
    def help(self, parser=None):
1905
 
        """
1906
 
        Prints a help message.
1907
 
 
1908
 
        :param parser: If supplied, the parser to use for generating help.  If \
1909
 
        not supplied, it is retrieved.
1910
 
        :type parser: cmdutil.CmdOptionParser
1911
 
        """
1912
 
        if parser==None:
1913
 
            parser=self.get_parser()
1914
 
        parser.print_help()
1915
 
        print """
1916
 
Performs a merge operation using the specified version.
1917
 
        """
1918
 
        return
1919
 
 
1920
 
class ELog(BaseCommand):
1921
 
    """
1922
 
    Produces a raw patchlog and invokes the user's editor
1923
 
    """
1924
 
    def __init__(self):
1925
 
        self.description="Edit a patchlog to commit"
1926
 
        try:
1927
 
            self.tree = arch.tree_root()
1928
 
        except:
1929
 
            self.tree = None
1930
 
 
1931
 
 
1932
 
    def do_command(self, cmdargs):
1933
 
        """
1934
 
        Master function that perfoms the "elog" command.
1935
 
        """
1936
 
        parser=self.get_parser()
1937
 
        (options, args) = parser.parse_args(cmdargs)
1938
 
        if self.tree is None:
1939
 
            raise arch.errors.TreeRootError
1940
 
 
1941
 
        try:
1942
 
            edit_log(self.tree, self.tree.tree_version)
1943
 
        except pylon.errors.NoEditorSpecified, e:
1944
 
            raise pylon.errors.CommandFailedWrapper(e)
1945
 
 
1946
 
    def get_parser(self):
1947
 
        """
1948
 
        Returns the options parser to use for the "merge" command.
1949
 
 
1950
 
        :rtype: cmdutil.CmdOptionParser
1951
 
        """
1952
 
        parser=cmdutil.CmdOptionParser("fai elog")
1953
 
        return parser 
1954
 
 
1955
 
 
1956
 
    def help(self, parser=None):
1957
 
        """
1958
 
        Invokes $EDITOR to produce a log for committing.
1959
 
 
1960
 
        :param parser: If supplied, the parser to use for generating help.  If \
1961
 
        not supplied, it is retrieved.
1962
 
        :type parser: cmdutil.CmdOptionParser
1963
 
        """
1964
 
        if parser==None:
1965
 
            parser=self.get_parser()
1966
 
        parser.print_help()
1967
 
        print """
1968
 
Invokes $EDITOR to produce a log for committing.
1969
 
        """
1970
 
        return
1971
 
 
1972
 
def edit_log(tree, version):
1973
 
    """Makes and edits the log for a tree.  Does all kinds of fancy things
1974
 
    like log templates and merge summaries and log-for-merge
1975
 
    
1976
 
    :param tree: The tree to edit the log for
1977
 
    :type tree: `arch.WorkingTree`
1978
 
    """
1979
 
    #ensure we have an editor before preparing the log
1980
 
    cmdutil.find_editor()
1981
 
    log = tree.log_message(create=False, version=version)
1982
 
    log_is_new = False
1983
 
    if log is None or cmdutil.prompt("Overwrite log"):
1984
 
        if log is not None:
1985
 
           os.remove(log.name)
1986
 
        log = tree.log_message(create=True, version=version)
1987
 
        log_is_new = True
1988
 
        tmplog = log.name
1989
 
        template = pylon.log_template_path(tree)
1990
 
        if template:
1991
 
            shutil.copyfile(template, tmplog)
1992
 
        comp_version = ancillary.comp_revision(tree).version
1993
 
        new_merges = cmdutil.iter_new_merges(tree, comp_version)
1994
 
        new_merges = cmdutil.direct_merges(new_merges)
1995
 
        log["Summary"] = pylon.merge_summary(new_merges, 
1996
 
                                         version)
1997
 
        if len(new_merges) > 0:   
1998
 
            if cmdutil.prompt("Log for merge"):
1999
 
                if cmdutil.prompt("changelog for merge"):
2000
 
                    mergestuff = "Patches applied:\n"
2001
 
                    mergestuff += pylon.changelog_for_merge(new_merges)
2002
 
                else:
2003
 
                    mergestuff = cmdutil.log_for_merge(tree, comp_version)
2004
 
                log.description += mergestuff
2005
 
        log.save()
2006
 
    try:
2007
 
        cmdutil.invoke_editor(log.name)
2008
 
    except:
2009
 
        if log_is_new:
2010
 
            os.remove(log.name)
2011
 
        raise
2012
 
 
2013
 
 
2014
 
class MirrorArchive(BaseCommand):
2015
 
    """
2016
 
    Updates a mirror from an archive
2017
 
    """
2018
 
    def __init__(self):
2019
 
        self.description="Update a mirror from an archive"
2020
 
 
2021
 
    def do_command(self, cmdargs):
2022
 
        """
2023
 
        Master function that perfoms the "revision" command.
2024
 
        """
2025
 
 
2026
 
        parser=self.get_parser()
2027
 
        (options, args) = parser.parse_args(cmdargs)
2028
 
        if len(args) > 1:
2029
 
            raise GetHelp
2030
 
        try:
2031
 
            tree = arch.tree_root()
2032
 
        except:
2033
 
            tree = None
2034
 
 
2035
 
        if len(args) == 0:
2036
 
            if tree is not None:
2037
 
                name = tree.tree_version()
2038
 
        else:
2039
 
            name = cmdutil.expand_alias(args[0], tree)
2040
 
            name = arch.NameParser(name)
2041
 
 
2042
 
        to_arch = name.get_archive()
2043
 
        from_arch = cmdutil.get_mirror_source(arch.Archive(to_arch))
2044
 
        limit = name.get_nonarch()
2045
 
 
2046
 
        iter = arch_core.mirror_archive(from_arch,to_arch, limit)
2047
 
        for line in arch.chatter_classifier(iter):
2048
 
            cmdutil.colorize(line)
2049
 
 
2050
 
    def get_parser(self):
2051
 
        """
2052
 
        Returns the options parser to use for the "revision" command.
2053
 
 
2054
 
        :rtype: cmdutil.CmdOptionParser
2055
 
        """
2056
 
        parser=cmdutil.CmdOptionParser("fai mirror-archive ARCHIVE")
2057
 
        return parser 
2058
 
 
2059
 
    def help(self, parser=None):
2060
 
        """
2061
 
        Prints a help message.
2062
 
 
2063
 
        :param parser: If supplied, the parser to use for generating help.  If \
2064
 
        not supplied, it is retrieved.
2065
 
        :type parser: cmdutil.CmdOptionParser
2066
 
        """
2067
 
        if parser==None:
2068
 
            parser=self.get_parser()
2069
 
        parser.print_help()
2070
 
        print """
2071
 
Updates a mirror from an archive.  If a branch, package, or version is
2072
 
supplied, only changes under it are mirrored.
2073
 
        """
2074
 
        return
2075
 
 
2076
 
def help_tree_spec():
2077
 
    print """Specifying revisions (default: tree)
2078
 
Revisions may be specified by alias, revision, version or patchlevel.
2079
 
Revisions or versions may be fully qualified.  Unqualified revisions, versions, 
2080
 
or patchlevels use the archive of the current project tree.  Versions will
2081
 
use the latest patchlevel in the tree.  Patchlevels will use the current tree-
2082
 
version.
2083
 
 
2084
 
Use "alias" to list available (user and automatic) aliases."""
2085
 
 
2086
 
auto_alias = [
2087
 
"acur", 
2088
 
"The latest revision in the archive of the tree-version.  You can specify \
2089
 
a different version like so: acur:foo--bar--0 (aliases can be used)",
2090
 
"tcur",
2091
 
"""(tree current) The latest revision in the tree of the tree-version. \
2092
 
You can specify a different version like so: tcur:foo--bar--0 (aliases can be \
2093
 
used).""",
2094
 
"tprev" , 
2095
 
"""(tree previous) The previous revision in the tree of the tree-version.  To \
2096
 
specify an older revision, use a number, e.g. "tprev:4" """,
2097
 
"tanc" , 
2098
 
"""(tree ancestor) The ancestor revision of the tree To specify an older \
2099
 
revision, use a number, e.g. "tanc:4".""",
2100
 
"tdate" , 
2101
 
"""(tree date) The latest revision from a given date, e.g. "tdate:July 6".""",
2102
 
"tmod" , 
2103
 
""" (tree modified) The latest revision to modify a given file, e.g. \
2104
 
"tmod:engine.cpp" or "tmod:engine.cpp:16".""",
2105
 
"ttag" , 
2106
 
"""(tree tag) The revision that was tagged into the current tree revision, \
2107
 
according to the tree""",
2108
 
"tagcur", 
2109
 
"""(tag current) The latest revision of the version that the current tree \
2110
 
was tagged from.""",
2111
 
"mergeanc" , 
2112
 
"""The common ancestor of the current tree and the specified revision. \
2113
 
Defaults to the first partner-version's latest revision or to tagcur.""",
2114
 
]
2115
 
 
2116
 
 
2117
 
def is_auto_alias(name):
2118
 
    """Determine whether a name is an auto alias name
2119
 
 
2120
 
    :param name: the name to check
2121
 
    :type name: str
2122
 
    :return: True if the name is an auto alias, false if not
2123
 
    :rtype: bool
2124
 
    """
2125
 
    return name in [f for (f, v) in pylon.util.iter_pairs(auto_alias)]
2126
 
 
2127
 
 
2128
 
def display_def(iter, wrap = 80):
2129
 
    """Display a list of definitions
2130
 
 
2131
 
    :param iter: iter of name, definition pairs
2132
 
    :type iter: iter of (str, str)
2133
 
    :param wrap: The width for text wrapping
2134
 
    :type wrap: int
2135
 
    """
2136
 
    vals = list(iter)
2137
 
    maxlen = 0
2138
 
    for (key, value) in vals:
2139
 
        if len(key) > maxlen:
2140
 
            maxlen = len(key)
2141
 
    for (key, value) in vals:
2142
 
        tw=textwrap.TextWrapper(width=wrap, 
2143
 
                                initial_indent=key.rjust(maxlen)+" : ",
2144
 
                                subsequent_indent="".rjust(maxlen+3))
2145
 
        print tw.fill(value)
2146
 
 
2147
 
 
2148
 
def help_aliases(tree):
2149
 
    print """Auto-generated aliases"""
2150
 
    display_def(pylon.util.iter_pairs(auto_alias))
2151
 
    print "User aliases"
2152
 
    display_def(ancillary.iter_all_alias(tree))
2153
 
 
2154
 
class Inventory(BaseCommand):
2155
 
    """List the status of files in the tree"""
2156
 
    def __init__(self):
2157
 
        self.description=self.__doc__
2158
 
 
2159
 
    def do_command(self, cmdargs):
2160
 
        """
2161
 
        Master function that perfoms the "revision" command.
2162
 
        """
2163
 
 
2164
 
        parser=self.get_parser()
2165
 
        (options, args) = parser.parse_args(cmdargs)
2166
 
        tree = arch.tree_root()
2167
 
        categories = []
2168
 
 
2169
 
        if (options.source):
2170
 
            categories.append(arch_core.SourceFile)
2171
 
        if (options.precious):
2172
 
            categories.append(arch_core.PreciousFile)
2173
 
        if (options.backup):
2174
 
            categories.append(arch_core.BackupFile)
2175
 
        if (options.junk):
2176
 
            categories.append(arch_core.JunkFile)
2177
 
 
2178
 
        if len(categories) == 1:
2179
 
            show_leading = False
2180
 
        else:
2181
 
            show_leading = True
2182
 
 
2183
 
        if len(categories) == 0:
2184
 
            categories = None
2185
 
 
2186
 
        if options.untagged:
2187
 
            categories = arch_core.non_root
2188
 
            show_leading = False
2189
 
            tagged = False
2190
 
        else:
2191
 
            tagged = None
2192
 
        
2193
 
        for file in arch_core.iter_inventory_filter(tree, None, 
2194
 
            control_files=options.control_files, 
2195
 
            categories = categories, tagged=tagged):
2196
 
            print arch_core.file_line(file, 
2197
 
                                      category = show_leading, 
2198
 
                                      untagged = show_leading,
2199
 
                                      id = options.ids)
2200
 
 
2201
 
    def get_parser(self):
2202
 
        """
2203
 
        Returns the options parser to use for the "revision" command.
2204
 
 
2205
 
        :rtype: cmdutil.CmdOptionParser
2206
 
        """
2207
 
        parser=cmdutil.CmdOptionParser("fai inventory [options]")
2208
 
        parser.add_option("--ids", action="store_true", dest="ids", 
2209
 
                          help="Show file ids")
2210
 
        parser.add_option("--control", action="store_true", 
2211
 
                          dest="control_files", help="include control files")
2212
 
        parser.add_option("--source", action="store_true", dest="source",
2213
 
                          help="List source files")
2214
 
        parser.add_option("--backup", action="store_true", dest="backup",
2215
 
                          help="List backup files")
2216
 
        parser.add_option("--precious", action="store_true", dest="precious",
2217
 
                          help="List precious files")
2218
 
        parser.add_option("--junk", action="store_true", dest="junk",
2219
 
                          help="List junk files")
2220
 
        parser.add_option("--unrecognized", action="store_true", 
2221
 
                          dest="unrecognized", help="List unrecognized files")
2222
 
        parser.add_option("--untagged", action="store_true", 
2223
 
                          dest="untagged", help="List only untagged files")
2224
 
        return parser 
2225
 
 
2226
 
    def help(self, parser=None):
2227
 
        """
2228
 
        Prints a help message.
2229
 
 
2230
 
        :param parser: If supplied, the parser to use for generating help.  If \
2231
 
        not supplied, it is retrieved.
2232
 
        :type parser: cmdutil.CmdOptionParser
2233
 
        """
2234
 
        if parser==None:
2235
 
            parser=self.get_parser()
2236
 
        parser.print_help()
2237
 
        print """
2238
 
Lists the status of files in the archive:
2239
 
S source
2240
 
P precious
2241
 
B backup
2242
 
J junk
2243
 
U unrecognized
2244
 
T tree root
2245
 
? untagged-source
2246
 
Leading letter are not displayed if only one kind of file is shown
2247
 
        """
2248
 
        return
2249
 
 
2250
 
 
2251
 
class Alias(BaseCommand):
2252
 
    """List or adjust aliases"""
2253
 
    def __init__(self):
2254
 
        self.description=self.__doc__
2255
 
 
2256
 
    def get_completer(self, arg, index):
2257
 
        if index > 2:
2258
 
            return ()
2259
 
        try:
2260
 
            self.tree = arch.tree_root()
2261
 
        except:
2262
 
            self.tree = None
2263
 
 
2264
 
        if index == 0:
2265
 
            return [part[0]+" " for part in ancillary.iter_all_alias(self.tree)]
2266
 
        elif index == 1:
2267
 
            return cmdutil.iter_revision_completions(arg, self.tree)
2268
 
 
2269
 
 
2270
 
    def do_command(self, cmdargs):
2271
 
        """
2272
 
        Master function that perfoms the "revision" command.
2273
 
        """
2274
 
 
2275
 
        parser=self.get_parser()
2276
 
        (options, args) = parser.parse_args(cmdargs)
2277
 
        try:
2278
 
            self.tree =  arch.tree_root()
2279
 
        except:
2280
 
            self.tree = None
2281
 
 
2282
 
 
2283
 
        try:
2284
 
            options.action(args, options)
2285
 
        except cmdutil.ForbiddenAliasSyntax, e:
2286
 
            raise CommandFailedWrapper(e)
2287
 
 
2288
 
    def no_prefix(self, alias):
2289
 
        if alias.startswith("^"):
2290
 
            alias = alias[1:]
2291
 
        return alias
2292
 
        
2293
 
    def arg_dispatch(self, args, options):
2294
 
        """Add, modify, or list aliases, depending on number of arguments
2295
 
 
2296
 
        :param args: The list of commandline arguments
2297
 
        :type args: list of str
2298
 
        :param options: The commandline options
2299
 
        """
2300
 
        if len(args) == 0:
2301
 
            help_aliases(self.tree)
2302
 
            return
2303
 
        else:
2304
 
            alias = self.no_prefix(args[0])
2305
 
            if len(args) == 1:
2306
 
                self.print_alias(alias)
2307
 
            elif (len(args)) == 2:
2308
 
                self.add(alias, args[1], options)
2309
 
            else:
2310
 
                raise cmdutil.GetHelp
2311
 
 
2312
 
    def print_alias(self, alias):
2313
 
        answer = None
2314
 
        if is_auto_alias(alias):
2315
 
            raise pylon.errors.IsAutoAlias(alias, "\"%s\" is an auto alias."
2316
 
                "  Use \"revision\" to expand auto aliases." % alias)
2317
 
        for pair in ancillary.iter_all_alias(self.tree):
2318
 
            if pair[0] == alias:
2319
 
                answer = pair[1]
2320
 
        if answer is not None:
2321
 
            print answer
2322
 
        else:
2323
 
            print "The alias %s is not assigned." % alias
2324
 
 
2325
 
    def add(self, alias, expansion, options):
2326
 
        """Add or modify aliases
2327
 
 
2328
 
        :param alias: The alias name to create/modify
2329
 
        :type alias: str
2330
 
        :param expansion: The expansion to assign to the alias name
2331
 
        :type expansion: str
2332
 
        :param options: The commandline options
2333
 
        """
2334
 
        if is_auto_alias(alias):
2335
 
            raise IsAutoAlias(alias)
2336
 
        newlist = ""
2337
 
        written = False
2338
 
        new_line = "%s=%s\n" % (alias, cmdutil.expand_alias(expansion, 
2339
 
            self.tree))
2340
 
        ancillary.check_alias(new_line.rstrip("\n"), [alias, expansion])
2341
 
 
2342
 
        for pair in self.get_iterator(options):
2343
 
            if pair[0] != alias:
2344
 
                newlist+="%s=%s\n" % (pair[0], pair[1])
2345
 
            elif not written:
2346
 
                newlist+=new_line
2347
 
                written = True
2348
 
        if not written:
2349
 
            newlist+=new_line
2350
 
        self.write_aliases(newlist, options)
2351
 
            
2352
 
    def delete(self, args, options):
2353
 
        """Delete the specified alias
2354
 
 
2355
 
        :param args: The list of arguments
2356
 
        :type args: list of str
2357
 
        :param options: The commandline options
2358
 
        """
2359
 
        deleted = False
2360
 
        if len(args) != 1:
2361
 
            raise cmdutil.GetHelp
2362
 
        alias = self.no_prefix(args[0])
2363
 
        if is_auto_alias(alias):
2364
 
            raise IsAutoAlias(alias)
2365
 
        newlist = ""
2366
 
        for pair in self.get_iterator(options):
2367
 
            if pair[0] != alias:
2368
 
                newlist+="%s=%s\n" % (pair[0], pair[1])
2369
 
            else:
2370
 
                deleted = True
2371
 
        if not deleted:
2372
 
            raise errors.NoSuchAlias(alias)
2373
 
        self.write_aliases(newlist, options)
2374
 
 
2375
 
    def get_alias_file(self, options):
2376
 
        """Return the name of the alias file to use
2377
 
 
2378
 
        :param options: The commandline options
2379
 
        """
2380
 
        if options.tree:
2381
 
            if self.tree is None:
2382
 
                self.tree == arch.tree_root()
2383
 
            return str(self.tree)+"/{arch}/+aliases"
2384
 
        else:
2385
 
            return "~/.aba/aliases"
2386
 
 
2387
 
    def get_iterator(self, options):
2388
 
        """Return the alias iterator to use
2389
 
 
2390
 
        :param options: The commandline options
2391
 
        """
2392
 
        return ancillary.iter_alias(self.get_alias_file(options))
2393
 
 
2394
 
    def write_aliases(self, newlist, options):
2395
 
        """Safely rewrite the alias file
2396
 
        :param newlist: The new list of aliases
2397
 
        :type newlist: str
2398
 
        :param options: The commandline options
2399
 
        """
2400
 
        filename = os.path.expanduser(self.get_alias_file(options))
2401
 
        file = util.NewFileVersion(filename)
2402
 
        file.write(newlist)
2403
 
        file.commit()
2404
 
 
2405
 
 
2406
 
    def get_parser(self):
2407
 
        """
2408
 
        Returns the options parser to use for the "alias" command.
2409
 
 
2410
 
        :rtype: cmdutil.CmdOptionParser
2411
 
        """
2412
 
        parser=cmdutil.CmdOptionParser("fai alias [ALIAS] [NAME]")
2413
 
        parser.add_option("-d", "--delete", action="store_const", dest="action",
2414
 
                          const=self.delete, default=self.arg_dispatch, 
2415
 
                          help="Delete an alias")
2416
 
        parser.add_option("--tree", action="store_true", dest="tree", 
2417
 
                          help="Create a per-tree alias", default=False)
2418
 
        return parser 
2419
 
 
2420
 
    def help(self, parser=None):
2421
 
        """
2422
 
        Prints a help message.
2423
 
 
2424
 
        :param parser: If supplied, the parser to use for generating help.  If \
2425
 
        not supplied, it is retrieved.
2426
 
        :type parser: cmdutil.CmdOptionParser
2427
 
        """
2428
 
        if parser==None:
2429
 
            parser=self.get_parser()
2430
 
        parser.print_help()
2431
 
        print """
2432
 
Lists current aliases or modifies the list of aliases.
2433
 
 
2434
 
If no arguments are supplied, aliases will be listed.  If two arguments are
2435
 
supplied, the specified alias will be created or modified.  If -d or --delete
2436
 
is supplied, the specified alias will be deleted.
2437
 
 
2438
 
You can create aliases that refer to any fully-qualified part of the
2439
 
Arch namespace, e.g. 
2440
 
archive, 
2441
 
archive/category, 
2442
 
archive/category--branch, 
2443
 
archive/category--branch--version (my favourite)
2444
 
archive/category--branch--version--patchlevel
2445
 
 
2446
 
Aliases can be used automatically by native commands.  To use them
2447
 
with external or tla commands, prefix them with ^ (you can do this
2448
 
with native commands, too).
2449
 
"""
2450
 
 
2451
 
 
2452
 
class RequestMerge(BaseCommand):
2453
 
    """Submit a merge request to Bug Goo"""
2454
 
    def __init__(self):
2455
 
        self.description=self.__doc__
2456
 
 
2457
 
    def do_command(self, cmdargs):
2458
 
        """Submit a merge request
2459
 
 
2460
 
        :param cmdargs: The commandline arguments
2461
 
        :type cmdargs: list of str
2462
 
        """
2463
 
        parser = self.get_parser()
2464
 
        (options, args) = parser.parse_args(cmdargs)
2465
 
        try:
2466
 
            cmdutil.find_editor()
2467
 
        except pylon.errors.NoEditorSpecified, e:
2468
 
            raise pylon.errors.CommandFailedWrapper(e)
2469
 
        try:
2470
 
            self.tree=arch.tree_root()
2471
 
        except:
2472
 
            self.tree=None
2473
 
        base, revisions = self.revision_specs(args)
2474
 
        message = self.make_headers(base, revisions)
2475
 
        message += self.make_summary(revisions)
2476
 
        path = self.edit_message(message)
2477
 
        message = self.tidy_message(path)
2478
 
        if cmdutil.prompt("Send merge"):
2479
 
            self.send_message(message)
2480
 
            print "Merge request sent"
2481
 
 
2482
 
    def make_headers(self, base, revisions):
2483
 
        """Produce email and Bug Goo header strings
2484
 
 
2485
 
        :param base: The base revision to apply merges to
2486
 
        :type base: `arch.Revision`
2487
 
        :param revisions: The revisions to replay into the base
2488
 
        :type revisions: list of `arch.Patchlog`
2489
 
        :return: The headers
2490
 
        :rtype: str
2491
 
        """
2492
 
        headers = "To: gnu-arch-users@gnu.org\n"
2493
 
        headers += "From: %s\n" % options.fromaddr
2494
 
        if len(revisions) == 1:
2495
 
            headers += "Subject: [MERGE REQUEST] %s\n" % revisions[0].summary
2496
 
        else:
2497
 
            headers += "Subject: [MERGE REQUEST]\n"
2498
 
        headers += "\n"
2499
 
        headers += "Base-Revision: %s\n" % base
2500
 
        for revision in revisions:
2501
 
            headers += "Revision: %s\n" % revision.revision
2502
 
        headers += "Bug: \n\n"
2503
 
        return headers
2504
 
 
2505
 
    def make_summary(self, logs):
2506
 
        """Generate a summary of merges
2507
 
 
2508
 
        :param logs: the patchlogs that were directly added by the merges
2509
 
        :type logs: list of `arch.Patchlog`
2510
 
        :return: the summary
2511
 
        :rtype: str
2512
 
        """ 
2513
 
        summary = ""
2514
 
        for log in logs:
2515
 
            summary+=str(log.revision)+"\n"
2516
 
            summary+=log.summary+"\n"
2517
 
            if log.description.strip():
2518
 
                summary+=log.description.strip('\n')+"\n\n"
2519
 
        return summary
2520
 
 
2521
 
    def revision_specs(self, args):
2522
 
        """Determine the base and merge revisions from tree and arguments.
2523
 
 
2524
 
        :param args: The parsed arguments
2525
 
        :type args: list of str
2526
 
        :return: The base revision and merge revisions 
2527
 
        :rtype: `arch.Revision`, list of `arch.Patchlog`
2528
 
        """
2529
 
        if len(args) > 0:
2530
 
            target_revision = cmdutil.determine_revision_arch(self.tree, 
2531
 
                                                              args[0])
2532
 
        else:
2533
 
            target_revision = arch_compound.tree_latest(self.tree)
2534
 
        if len(args) > 1:
2535
 
            merges = [ arch.Patchlog(cmdutil.determine_revision_arch(
2536
 
                       self.tree, f)) for f in args[1:] ]
2537
 
        else:
2538
 
            if self.tree is None:
2539
 
                raise CantDetermineRevision("", "Not in a project tree")
2540
 
            merge_iter = cmdutil.iter_new_merges(self.tree, 
2541
 
                                                 target_revision.version, 
2542
 
                                                 False)
2543
 
            merges = [f for f in cmdutil.direct_merges(merge_iter)]
2544
 
        return (target_revision, merges)
2545
 
 
2546
 
    def edit_message(self, message):
2547
 
        """Edit an email message in the user's standard editor
2548
 
 
2549
 
        :param message: The message to edit
2550
 
        :type message: str
2551
 
        :return: the path of the edited message
2552
 
        :rtype: str
2553
 
        """
2554
 
        if self.tree is None:
2555
 
            path = os.get_cwd()
2556
 
        else:
2557
 
            path = self.tree
2558
 
        path += "/,merge-request"
2559
 
        file = open(path, 'w')
2560
 
        file.write(message)
2561
 
        file.flush()
2562
 
        cmdutil.invoke_editor(path)
2563
 
        return path
2564
 
 
2565
 
    def tidy_message(self, path):
2566
 
        """Validate and clean up message.
2567
 
 
2568
 
        :param path: The path to the message to clean up
2569
 
        :type path: str
2570
 
        :return: The parsed message
2571
 
        :rtype: `email.Message`
2572
 
        """
2573
 
        mail = email.message_from_file(open(path))
2574
 
        if mail["Subject"].strip() == "[MERGE REQUEST]":
2575
 
            raise BlandSubject
2576
 
        
2577
 
        request = email.message_from_string(mail.get_payload())
2578
 
        if request.has_key("Bug"):
2579
 
            if request["Bug"].strip()=="":
2580
 
                del request["Bug"]
2581
 
        mail.set_payload(request.as_string())
2582
 
        return mail
2583
 
 
2584
 
    def send_message(self, message):
2585
 
        """Send a message, using its headers to address it.
2586
 
 
2587
 
        :param message: The message to send
2588
 
        :type message: `email.Message`"""
2589
 
        server = smtplib.SMTP("localhost")
2590
 
        server.sendmail(message['From'], message['To'], message.as_string())
2591
 
        server.quit()
2592
 
 
2593
 
    def help(self, parser=None):
2594
 
        """Print a usage message
2595
 
 
2596
 
        :param parser: The options parser to use
2597
 
        :type parser: `cmdutil.CmdOptionParser`
2598
 
        """
2599
 
        if parser is None:
2600
 
            parser = self.get_parser()
2601
 
        parser.print_help()
2602
 
        print """
2603
 
Sends a merge request formatted for Bug Goo.  Intended use: get the tree
2604
 
you'd like to merge into.  Apply the merges you want.  Invoke request-merge.
2605
 
The merge request will open in your $EDITOR.
2606
 
 
2607
 
When no TARGET is specified, it uses the current tree revision.  When
2608
 
no MERGE is specified, it uses the direct merges (as in "revisions
2609
 
--direct-merges").  But you can specify just the TARGET, or all the MERGE
2610
 
revisions.
2611
 
"""
2612
 
 
2613
 
    def get_parser(self):
2614
 
        """Produce a commandline parser for this command.
2615
 
 
2616
 
        :rtype: `cmdutil.CmdOptionParser`
2617
 
        """
2618
 
        parser=cmdutil.CmdOptionParser("request-merge [TARGET] [MERGE1...]")
2619
 
        return parser
2620
 
 
2621
 
commands = { 
2622
 
'changes' : Changes,
2623
 
'help' : Help,
2624
 
'update': Update,
2625
 
'apply-changes':ApplyChanges,
2626
 
'cat-log': CatLog,
2627
 
'commit': Commit,
2628
 
'revision': Revision,
2629
 
'revisions': Revisions,
2630
 
'get': Get,
2631
 
'revert': Revert,
2632
 
'shell': Shell,
2633
 
'add-id': AddID,
2634
 
'merge': Merge,
2635
 
'elog': ELog,
2636
 
'mirror-archive': MirrorArchive,
2637
 
'ninventory': Inventory,
2638
 
'alias' : Alias,
2639
 
'request-merge': RequestMerge,
2640
 
}
2641
 
 
2642
 
def my_import(mod_name):
2643
 
    module = __import__(mod_name)
2644
 
    components = mod_name.split('.')
2645
 
    for comp in components[1:]:
2646
 
        module = getattr(module, comp)
2647
 
    return module
2648
 
 
2649
 
def plugin(mod_name):
2650
 
    module = my_import(mod_name)
2651
 
    module.add_command(commands)
2652
 
 
2653
 
for file in os.listdir(sys.path[0]+"/command"):
2654
 
    if len(file) > 3 and file[-3:] == ".py" and file != "__init__.py":
2655
 
        plugin("command."+file[:-3])
2656
 
 
2657
 
suggestions = {
2658
 
'apply-delta' : "Try \"apply-changes\".",
2659
 
'delta' : "To compare two revisions, use \"changes\".",
2660
 
'diff-rev' : "To compare two revisions, use \"changes\".",
2661
 
'undo' : "To undo local changes, use \"revert\".",
2662
 
'undelete' : "To undo only deletions, use \"revert --deletions\"",
2663
 
'missing-from' : "Try \"revisions --missing-from\".",
2664
 
'missing' : "Try \"revisions --missing\".",
2665
 
'missing-merge' : "Try \"revisions --partner-missing\".",
2666
 
'new-merges' : "Try \"revisions --new-merges\".",
2667
 
'cachedrevs' : "Try \"revisions --cacherevs\". (no 'd')",
2668
 
'logs' : "Try \"revisions --logs\"",
2669
 
'tree-source' : "Use the \"^ttag\" alias (\"revision ^ttag\")",
2670
 
'latest-revision' : "Use the \"^acur\" alias (\"revision ^acur\")",
2671
 
'change-version' : "Try \"update REVISION\"",
2672
 
'tree-revision' : "Use the \"^tcur\" alias (\"revision ^tcur\")",
2673
 
'rev-depends' : "Use revisions --dependencies",
2674
 
'auto-get' : "Plain get will do archive lookups",
2675
 
'tagline' : "Use add-id.  It uses taglines in tagline trees",
2676
 
'emlog' : "Use elog.  It automatically adds log-for-merge text, if any",
2677
 
'library-revisions' : "Use revisions --library",
2678
 
'file-revert' : "Use revert FILE",
2679
 
'join-branch' : "Use replay --logs-only"
2680
 
}
2681
 
# arch-tag: 19d5739d-3708-486c-93ba-deecc3027fc7