85
by Aaron Bentley
Added annotate plugin |
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
|