19
19
# along with this program. If not, see <http://www.gnu.org/licenses/>.
22
import os, sys, getopt
24
from vcs.vcs import Vcs
30
29
def __init__( self, version ):
31
self.name = os.path.basename( sys.argv[ 0 ] )
32
30
self.version = version
35
def die( self, error_message ):
36
prefix = self.name + ( ' ' + the.command if the.command else '' )
37
print >> sys.stderr, '%s: %s' % ( prefix, error_message )
31
self.program = os.path.basename( sys.argv[ 0 ] )
34
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
def print_usage( self, error_message ):
42
command = ' ' + the.command if the.command else ''
43
self.die( error_message + \
44
"\nTry '%s%s --help' for more information." % \
45
( self.name, command ) )
48
40
def print_help( self ):
49
print "Usage: " + self.name + " COMMAND [OPTION]..."
41
print "Usage: " + self.program + " COMMAND [OPTION]..."
51
43
# 01234567890123456789012345678901234567890123456789012345678901234567890123456789
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)."
44
print "Manage your home directories, across multiple computers, similar to how you"
45
print "would software under version control."
56
47
print "Global options (for all commands):"
57
48
print " --help display help and exit"
61
52
print " init initialise a local copy of your repositories"
62
53
print " update update files in your home directory"
63
54
print " resolve try to finish an update (that had conflicts)"
64
# print " add add local files/changes to the repository"
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"
68
# print " revert undo changes made to a local file"
69
# print " stage-add stage local files/changes"
70
# print " stage-remove stage the removal of files"
55
print " add add local files/changes to the repository"
56
print " remove remove a local file from the repository"
57
print " revert undo changes made to a local file"
58
print " stage-add stage local files/changes"
59
print " stage-remove stage the removal of files"
71
60
print " stage-revert revert staged changes"
72
# print " stage-status show status of staging area"
73
# print " stage-diff shows staged changes"
74
# print " stage-commit commit staged changes to repository"
61
print " stage-status show status of staging area"
62
print " stage-commit commit staged changes to repository"
76
64
print "For help about a particular command (including the additional options that the"
77
65
print "command accepts) try typing:"
78
print " $ " + self.name + " COMMAND --help"
66
print " $ " + self.program + " COMMAND --help"
82
70
def print_version( self ):
83
71
print "stdhome " + self.version
85
print "Copyright (C) 2013 to 2014 Tim Marston"
73
print "Copyright (C) 2013 Tim Marston"
87
75
# 01234567890123456789012345678901234567890123456789012345678901234567890123456789
88
76
print "This program is free software, and you may use, modify and redistribute it"
104
92
- `command`: the given command
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:
95
if [ 'init', 'update', 'resolve', 'add', 'remove', 'revert',
96
'stage-add', 'stage-remove', 'stage-revert', 'stage-status',
97
'stage-commit' ].count( command ) == 1:
119
}.get( command, False )
120
if alias: return alias
101
elif command == 'up':
103
elif command == 'rm':
126
111
def get_command_argument( self, args ):
148
132
sys.argv[ 1: ], "",
149
133
[ "help", "version" ] )
151
for opt, optarg in opts:
135
for opt, optargs in opts:
152
136
# we only show help if there are no non-option arguments (e.g.,
153
137
# a command) specified. If a command has been specified it will
154
138
# have to be parsed and --help will be handled by it, instead)
158
142
self.print_version()
160
144
except getopt.GetoptError as e:
161
# ignore errors -- we aren't parsing the command line properly yet
145
# ignore any errors -- we aren't parsing the command line properly
164
148
# find the first non-option argument (the command)
165
the.command = self.get_command_argument( sys.argv[ 1: ] )
166
if the.command == None:
149
self.command = self.get_command_argument( sys.argv[ 1: ] )
150
if self.command == None:
167
151
self.print_usage( "missing command" )
169
153
# check command is valid
170
the.command = self.check_command( the.command )
171
if the.command == None:
154
self.command = self.check_command( self.command )
155
if self.command == None:
172
156
self.print_usage( "bad command" )
174
# calculate module and class name
175
class_name = module_name = ''
176
bits = the.command.split( '-' )
158
# calculate class name
159
bits = self.command.split( '-' )
160
class_name = 'Command'
178
162
class_name += bit[ 0 ].upper() + bit[ 1 : ]
179
if module_name: module_name += '_'
181
class_name += 'Command'
183
# import module and instantiate the class
184
module = __import__( 'stdhome.command.' + module_name,
185
fromlist = [ class_name ] )
186
instance = getattr( module, class_name )()
190
instance.parse_command_line()
192
except( getopt.GetoptError, self.UsageError ) as e:
193
self.print_usage( e.msg )
194
except Vcs.VcsError as e:
195
message = e.msg.rstrip()
197
message += '\n\nOUTPUT:\n' + e.output.rstrip()
199
except self.FatalError as e:
203
class UsageError( Exception ):
205
def __init__( self, error_message ):
206
self.msg = error_message
209
class FatalError( Exception ):
211
def __init__( self, message ):
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()