/stdhome

To get this branch, use:
bzr branch http://bzr.ed.am/stdhome

« back to all changes in this revision

Viewing changes to lib/stdhome/program.py

  • Committer: Tim Marston
  • Date: 2014-03-08 00:47:23 UTC
  • Revision ID: tim@ed.am-20140308004723-hkl3s2hobsblf72o
added diff command; moved all command to commands subdir; made stage-revert
handle ongoing deployment automatically (now that initial revno is known); made
verbose level incremental; detect obstructing conflicts in ConflictWalker;
handle files deleted from repo during copy-out (update)

Show diffs side-by-side

added added

removed removed

21
21
 
22
22
import os, sys, getopt
23
23
import the
 
24
from vcs.vcs import Vcs
24
25
 
25
26
 
26
27
class Program:
38
39
 
39
40
 
40
41
        def print_usage( self, error_message ):
41
 
                suffix = ' ' + the.command if the.command else ''
 
42
                command = ' ' + the.command if the.command else ''
42
43
                self.die( error_message + \
43
44
                                  "\nTry '%s%s --help' for more information." % \
44
 
                                  ( self.name, suffix ) )
 
45
                                  ( self.name, command ) )
45
46
 
46
47
 
47
48
        def print_help( self ):
48
49
                print "Usage: " + self.name + " COMMAND [OPTION]..."
49
50
                print
50
51
                #      01234567890123456789012345678901234567890123456789012345678901234567890123456789
51
 
                print "Manage your home directories, across multiple computers, similar to how you"
52
 
                print "would software under version control."
 
52
                print "Tool to manage a set of files in your home directory and distribute them across"
 
53
                print "multiple computers, merging local changes (in the same way as you would manage"
 
54
                print "source code under version control)."
53
55
                print
54
56
                print "Global options (for all commands):"
55
57
                print "     --help     display help and exit"
61
63
                print "  resolve       try to finish an update (that had conflicts)"
62
64
                print "  add           add local files/changes to the repository"
63
65
                print "  remove        remove a local file from the repository"
 
66
                print "  status        list files that have changed locally"
 
67
                print "  diff          shows changes made to local files"
64
68
                print "  revert        undo changes made to a local file"
65
69
                print "  stage-add     stage local files/changes"
66
70
                print "  stage-remove  stage the removal of files"
67
71
                print "  stage-revert  revert staged changes"
68
72
                print "  stage-status  show status of staging area"
 
73
                print "  stage-diff    shows staged changes"
69
74
                print "  stage-commit  commit staged changes to repository"
70
75
                print
71
76
                print "For help about a particular command (including the additional options that the"
77
82
        def print_version( self ):
78
83
                print "stdhome " + self.version
79
84
                print
80
 
                print "Copyright (C) 2013 Tim Marston"
 
85
                print "Copyright (C) 2013 to 2014 Tim Marston"
81
86
                print
82
87
                #      01234567890123456789012345678901234567890123456789012345678901234567890123456789
83
88
                print "This program is free software, and you may use, modify and redistribute it"
99
104
                - `command`: the given command
100
105
                """
101
106
                # commands
102
 
                if [ 'init', 'update', 'resolve', 'add', 'remove', 'revert',
103
 
                         'stage-add', 'stage-remove', 'stage-revert', 'stage-status',
104
 
                         'stage-commit' ].count( command ) == 1:
 
107
                if [ 'init', 'update', 'resolve', 'add', 'remove', 'revert', 'status',
 
108
                         'diff', 'stage-add', 'stage-remove', 'stage-revert',
 
109
                         'stage-status', 'stage-diff', 'stage-commit'
 
110
                ].count( command ) == 1:
105
111
                        return command
106
112
 
107
113
                # resolve aliases
108
 
                elif command == 'up':
109
 
                        return 'update'
110
 
                elif command == 'rm':
111
 
                        return 'remove'
 
114
                alias = {
 
115
                        'up': 'update',
 
116
                        'rm': 'remove',
 
117
                        'st': 'status',
 
118
                        'co': 'init',
 
119
                }.get( command, False )
 
120
                if alias: return alias
112
121
 
113
122
                # invalid
114
 
                else:
115
 
                        return None
 
123
                return None
116
124
 
117
125
 
118
126
        def get_command_argument( self, args ):
163
171
                if the.command == None:
164
172
                        self.print_usage( "bad command" )
165
173
 
166
 
                # calculate class name
 
174
                # calculate module and class name
 
175
                class_name = module_name = ''
167
176
                bits = the.command.split( '-' )
168
 
                class_name = 'Command'
169
177
                for bit in bits:
170
178
                        class_name += bit[ 0 ].upper() + bit[ 1 : ]
 
179
                        if module_name: module_name += '_'
 
180
                        module_name += bit
 
181
                class_name += 'Command'
171
182
 
172
 
                # instantiate the command class
173
 
                module = __import__( 'stdhome.command_' + the.command,
 
183
                # import module and instantiate the class
 
184
                module = __import__( 'stdhome.command.' + module_name,
174
185
                                                         fromlist = [ class_name ] )
175
186
                instance = getattr( module, class_name )()
176
187
 
177
 
                # run it
 
188
                # run the command
178
189
                try:
179
190
                        instance.parse_command_line()
180
191
                        instance.run()
181
 
                except ( getopt.GetoptError, self.UsageError ) as e:
 
192
                except( getopt.GetoptError, self.UsageError ) as e:
182
193
                        self.print_usage( e.msg )
 
194
                except Vcs.VcsError as e:
 
195
                        message = e.msg.rstrip()
 
196
                        if the.verbose:
 
197
                                message += '\n\nOUTPUT:\n' + e.output.rstrip()
 
198
                        self.die( message )
183
199
                except self.FatalError as e:
184
 
                        message = e.msg
185
 
                        if the.verbose and e.output:
186
 
                                message += '\n\nOUTPUT:\n' + e.output
187
 
                        self.die( message )
 
200
                        self.die( e.msg )
188
201
 
189
202
 
190
203
        class UsageError( Exception ):
195
208
 
196
209
        class FatalError( Exception ):
197
210
 
198
 
                def __init__( self, message, output = None ):
 
211
                def __init__( self, message ):
199
212
                        self.msg = message
200
 
                        self.output = output