19
19
# along with this program. If not, see <http://www.gnu.org/licenses/>.
22
import os, sys, getopt
29
29
def __init__( self, version ):
30
self.name = os.path.basename( sys.argv[ 0 ] )
30
31
self.version = version
31
self.program = os.path.basename( sys.argv[ 0 ] )
34
def die( self, error_message ):
35
prefix = self.name + ( ' ' + the.command if the.command else '' )
36
print >> sys.stderr, '%s: %s' % ( prefix, error_message )
34
40
def print_usage( self, error_message ):
35
print >> sys.stderr, self.program + ": " + error_message
36
print >> sys.stderr, "Try '" + self.program + " --help' for more information."
41
command = ' ' + the.command if the.command else ''
42
self.die( error_message + \
43
"\nTry '%s%s --help' for more information." % \
44
( self.name, command ) )
40
47
def print_help( self ):
41
print "Usage: " + self.program + " COMMAND [OPTION]..."
48
print "Usage: " + self.name + " COMMAND [OPTION]..."
43
50
# 01234567890123456789012345678901234567890123456789012345678901234567890123456789
44
print "Manage your home directories, across multiple computers, similar to how you"
45
print "would software under version control."
51
print "Tool to manage a set of files in your home directory and distribute them across"
52
print "multiple computers, merging local changes (in the same way as you would manage"
53
print "source code under version control)."
47
55
print "Global options (for all commands):"
48
56
print " --help display help and exit"
54
62
print " resolve try to finish an update (that had conflicts)"
55
63
print " add add local files/changes to the repository"
56
64
print " remove remove a local file from the repository"
65
print " status list files that have changed locally"
66
print " diff shows changes made to local files"
57
67
print " revert undo changes made to a local file"
58
68
print " stage-add stage local files/changes"
59
69
print " stage-remove stage the removal of files"
60
70
print " stage-revert revert staged changes"
61
71
print " stage-status show status of staging area"
72
print " stage-diff shows staged changes"
62
73
print " stage-commit commit staged changes to repository"
64
75
print "For help about a particular command (including the additional options that the"
65
76
print "command accepts) try typing:"
66
print " $ " + self.program + " COMMAND --help"
77
print " $ " + self.name + " COMMAND --help"
92
103
- `command`: the given command
95
if [ 'init', 'update', 'resolve', 'add', 'remove', 'revert',
96
'stage-add', 'stage-remove', 'stage-revert', 'stage-status',
97
'stage-commit' ].count( command ) == 1:
106
if [ 'init', 'update', 'resolve', 'add', 'remove', 'revert', 'status',
107
'diff', 'stage-add', 'stage-remove', 'stage-revert',
108
'stage-status', 'stage-diff', 'stage-commit'
109
].count( command ) == 1:
101
113
elif command == 'up':
103
115
elif command == 'rm':
132
145
sys.argv[ 1: ], "",
133
146
[ "help", "version" ] )
135
for opt, optargs in opts:
148
for opt, optarg in opts:
136
149
# we only show help if there are no non-option arguments (e.g.,
137
150
# a command) specified. If a command has been specified it will
138
151
# have to be parsed and --help will be handled by it, instead)
142
155
self.print_version()
144
157
except getopt.GetoptError as e:
145
# ignore any errors -- we aren't parsing the command line properly
158
# ignore errors -- we aren't parsing the command line properly yet
148
161
# find the first non-option argument (the command)
149
self.command = self.get_command_argument( sys.argv[ 1: ] )
150
if self.command == None:
162
the.command = self.get_command_argument( sys.argv[ 1: ] )
163
if the.command == None:
151
164
self.print_usage( "missing command" )
153
166
# check command is valid
154
self.command = self.check_command( self.command )
155
if self.command == None:
167
the.command = self.check_command( the.command )
168
if the.command == None:
156
169
self.print_usage( "bad command" )
158
# calculate class name
159
bits = self.command.split( '-' )
171
# calculate module and class name
172
bits = the.command.split( '-' )
160
173
class_name = 'Command'
174
module_name = 'command'
162
176
class_name += bit[ 0 ].upper() + bit[ 1 : ]
164
# instantiate and run the command class
165
module = __import__( 'stdhome.command_' + self.command )
166
the_class = getattr( module, class_name )
167
instance = the_class()
177
module_name += '_' + bit
179
# import module and instantiate the class
180
module = __import__( 'stdhome.' + module_name,
181
fromlist = [ class_name ] )
182
instance = getattr( module, class_name )()
186
instance.parse_command_line()
188
except ( getopt.GetoptError, self.UsageError ) as e:
189
self.print_usage( e.msg )
190
except self.FatalError as e:
191
message = e.msg.rstrip()
192
if the.verbose and hasattr( e, 'output' ) and e.output:
193
message += '\n\nOUTPUT:\n' + e.output.rstrip()
197
class UsageError( Exception ):
199
def __init__( self, error_message ):
200
self.msg = error_message
203
class FatalError( Exception ):
205
def __init__( self, message, output = None ):