9 output
= fd
.readlines()
11 return [x
.strip() for x
in output
]
14 rc
= os
.system("%s > /dev/null 2>&1" % cmd
)
18 class YapError(Exception):
19 def __init__(self
, msg
):
25 def takes_options(options
):
27 func
.options
= options
32 def _add_new_file(self
, file):
33 repo
= get_output('git rev-parse --git-dir')[0]
34 dir = os
.path
.join(repo
, 'yap')
39 files
= self
._get
_new
_files
()
41 path
= os
.path
.join(dir, 'new-files')
42 pickle
.dump(files
, open(path
, 'w'))
44 def _get_new_files(self
):
45 repo
= get_output('git rev-parse --git-dir')[0]
46 path
= os
.path
.join(repo
, 'yap', 'new-files')
48 files
= pickle
.load(file(path
))
55 if get_output("git ls-files --cached '%s'" % f
) != []:
60 def _remove_new_file(self
, file):
61 files
= self
._get
_new
_files
()
62 files
= filter(lambda x
: x
!= file, files
)
64 repo
= get_output('git rev-parse --git-dir')[0]
65 path
= os
.path
.join(repo
, 'yap', 'new-files')
66 pickle
.dump(files
, open(path
, 'w'))
68 def _clear_new_files(self
):
69 repo
= get_output('git rev-parse --git-dir')[0]
70 path
= os
.path
.join(repo
, 'yap', 'new-files')
73 def _assert_file_exists(self
, file):
74 if not os
.access(file, os
.R_OK
):
75 raise YapError("No such file: %s" % file)
77 def _get_staged_files(self
):
78 if run_command("git rev-parse HEAD"):
79 files
= get_output("git ls-files --cached")
81 files
= get_output("git diff-index --cached --name-only HEAD")
84 def _get_unstaged_files(self
):
85 files
= self
._get
_new
_files
()
86 files
+= get_output("git ls-files -m")
89 def _delete_branch(self
, branch
, force
):
90 current
= get_output("git symbolic-ref HEAD")[0]
91 current
= current
.replace('refs/heads/', '')
93 raise YapError("Can't delete current branch")
95 ref
= get_output("git rev-parse 'refs/heads/%s'" % branch
)
97 raise YapError("No such branch: %s" % branch
)
98 os
.system("git update-ref -d 'refs/heads/%s' '%s'" % (branch
, ref
[0]))
101 name
= get_output("git name-rev --name-only '%s'" % ref
[0])[0]
102 if name
== 'undefined':
103 os
.system("git update-ref 'refs/heads/%s' '%s'" % (branch
, ref
[0]))
104 raise YapError("Refusing to delete leaf branch (use -f to force)")
105 def _get_pager_cmd(self
):
106 if 'YAP_PAGER' in os
.environ
:
107 return os
.environ
['YAP_PAGER']
108 elif 'GIT_PAGER' in os
.environ
:
109 return os
.environ
['GIT_PAGER']
110 elif 'PAGER' in os
.environ
:
111 return os
.environ
['PAGER']
115 def _add_one(self
, file):
116 self
._assert
_file
_exists
(file)
117 x
= get_output("git ls-files '%s'" % file)
119 raise YapError("File '%s' already in repository" % file)
120 self
._add
_new
_file
(file)
122 def _rm_one(self
, file):
123 self
._assert
_file
_exists
(file)
124 if get_output("git ls-files '%s'" % file) != []:
125 os
.system("git rm --cached '%s'" % file)
126 self
._remove
_new
_file
(file)
128 def _stage_one(self
, file):
129 self
._assert
_file
_exists
(file)
130 os
.system("git update-index --add '%s'" % file)
132 def _unstage_one(self
, file):
133 self
._assert
_file
_exists
(file)
134 if run_command("git rev-parse HEAD"):
135 os
.system("git update-index --force-remove '%s'" % file)
137 os
.system("git diff-index -p HEAD '%s' | git apply -R --cached" % file)
139 def _revert_one(self
, file):
140 self
._assert
_file
_exists
(file)
141 os
.system("git checkout-index -f '%s'" % file)
143 def _parse_commit(self
, commit
):
144 lines
= get_output("git cat-file commit '%s'" % commit
)
149 if mode
!= 'commit' and l
.strip() == "":
154 commit
['log'].append(l
)
161 commit
['log'] = '\n'.join(commit
['log'])
164 def cmd_clone(self
, url
, directory
=""):
166 # XXX: implement in terms of init + remote add + fetch
167 os
.system("git clone '%s' %s" % (url
, directory
))
170 os
.system("git init")
172 def cmd_add(self
, *files
):
181 def cmd_rm(self
, *files
):
190 def cmd_stage(self
, *files
):
199 def cmd_unstage(self
, *files
):
208 def cmd_status(self
):
209 branch
= get_output("git symbolic-ref HEAD")[0]
210 branch
= branch
.replace('refs/heads/', '')
211 print "Current branch: %s" % branch
213 print "Files with staged changes:"
214 files
= self
._get
_staged
_files
()
220 print "Files with unstaged changes:"
221 prefix
= get_output("git rev-parse --show-prefix")
222 files
= self
._get
_unstaged
_files
()
225 f
= os
.path
.join(prefix
[0], f
)
231 def cmd_revert(self
, *files
, **flags
):
234 os
.system("git checkout-index -f -a")
245 def cmd_commit(self
, **flags
):
246 if '-a' in flags
and '-d' in flags
:
247 raise YapError("Conflicting flags: -a and -d")
249 if '-d' not in flags
and self
._get
_unstaged
_files
():
250 if '-a' not in flags
and self
._get
_staged
_files
():
251 raise YapError("Staged and unstaged changes present. Specify what to commit")
252 os
.system("git diff-files -p | git apply --cached 2>/dev/null")
253 for f
in self
._get
_new
_files
():
256 if not self
._get
_staged
_files
():
257 raise YapError("No changes to commit")
259 tree
= get_output("git write-tree")[0]
261 parent
= get_output("git rev-parse HEAD 2> /dev/null")[0]
263 if os
.environ
.has_key('YAP_EDITOR'):
264 editor
= os
.environ
['YAP_EDITOR']
265 elif os
.environ
.has_key('GIT_EDITOR'):
266 editor
= os
.environ
['GIT_EDITOR']
267 elif os
.environ
.has_key('EDITOR'):
268 editor
= os
.environ
['EDITOR']
272 fd
, tmpfile
= tempfile
.mkstemp("yap")
274 if os
.system("%s '%s'" % (editor
, tmpfile
)) != 0:
275 raise YapError("Editing commit message failed")
277 commit
= get_output("git commit-tree '%s' -p '%s' < '%s'" % (tree
, parent
, tmpfile
))
279 commit
= get_output("git commit-tree '%s' < '%s'" % (tree
, tmpfile
))
281 raise YapError("Commit failed; no log message?")
283 os
.system("git update-ref HEAD '%s'" % commit
[0])
286 def cmd_uncommit(self
):
287 commit
= self
._parse
_commit
("HEAD")
288 repo
= get_output('git rev-parse --git-dir')[0]
289 dir = os
.path
.join(repo
, 'yap')
294 msg_file
= os
.path
.join(dir, 'msg')
295 fd
= file(msg_file
, 'w')
296 print >>fd
, commit
['log']
299 tree
= get_output("git rev-parse HEAD^")
300 os
.system("git update-ref -m uncommit HEAD '%s'" % tree
[0])
303 def cmd_version(self
):
304 print "Yap version 0.1"
307 def cmd_log(self
, *paths
, **flags
):
308 "[-r <rev>] <path>..."
309 rev
= flags
.get('-r', 'HEAD')
310 paths
= ' '.join(paths
)
311 os
.system("git log --name-status '%s' -- %s" % (rev
, paths
))
314 def cmd_diff(self
, **flags
):
316 if '-u' in flags
and '-d' in flags
:
317 raise YapError("Conflicting flags: -u and -d")
319 pager
= self
._get
_pager
_cmd
()
321 os
.system("git update-index -q --refresh")
323 os
.system("git diff-files -p | %s" % pager
)
325 os
.system("git diff-index --cached -p HEAD | %s" % pager
)
327 os
.system("git diff-index -p HEAD | %s" % pager
)
329 @takes_options("fd:")
330 def cmd_branch(self
, branch
=None, **flags
):
331 "[ [-f] -d <branch> | <branch> ]"
332 force
= '-f' in flags
334 self
._delete
_branch
(flags
['-d'], force
)
338 if branch
is not None:
339 ref
= get_output("git rev-parse HEAD")
341 raise YapError("No branch point yet. Make a commit")
342 os
.system("git update-ref 'refs/heads/%s' '%s'" % (branch
, ref
[0]))
344 current
= get_output("git symbolic-ref HEAD")[0]
345 branches
= get_output("git for-each-ref --format='%(refname)' 'refs/heads/*'")
351 b
= b
.replace('refs/heads/', '')
354 def cmd_switch(self
, branch
):
356 ref
= get_output("git rev-parse 'refs/heads/%s'" % branch
)
358 raise YapError("No such branch: %s" % branch
)
360 # XXX: support merging like git-checkout
361 if self
._get
_unstaged
_files
() or self
._get
_staged
_files
():
362 raise YapError("You have uncommitted changes. Commit them first")
364 os
.system("git symbolic-ref HEAD refs/heads/'%s'" % branch
)
365 os
.system("git read-tree HEAD")
366 os
.system("git checkout-index -f -a")
370 def cmd_point(self
, where
, **flags
):
372 head
= get_output("git rev-parse HEAD")
374 raise YapError("No commit yet; nowhere to point")
376 ref
= get_output("git rev-parse '%s'" % where
)
378 raise YapError("Not a valid ref: %s" % where
)
380 if self
._get
_unstaged
_files
() or self
._get
_staged
_files
():
381 raise YapError("You have uncommitted changes. Commit them first")
383 type = get_output("git cat-file -t '%s'" % ref
[0])
384 if type and type[0] == "tag":
385 tag
= get_output("git cat-file tag '%s'" % ref
[0])
386 ref
[0] = tag
[0].split(' ')[1]
388 os
.system("git update-ref HEAD '%s'" % ref
[0])
390 if '-f' not in flags
:
391 name
= get_output("git name-rev --name-only '%s'" % head
[0])[0]
392 if name
== "undefined":
393 os
.system("git update-ref HEAD '%s'" % head
[0])
394 raise YapError("Pointing there will lose commits. Use -f to force")
396 os
.system("git read-tree HEAD")
397 os
.system("git checkout-index -f -a")
398 os
.system("git update-index --refresh")
400 def cmd_history(self
, subcmd
, commit
):
401 "amend | drop <commit>"
403 if subcmd
not in ("amend", "drop"):
406 # XXX: ensure no rebase in progress
408 if subcmd
== "amend":
409 # XXX: Use cmd_commit rules
410 stash
= get_output("git stash create")
411 os
.system("git reset --hard")
413 raise YapError("Failed to stash; no changes?")
415 fd
, tmpfile
= tempfile
.mkstemp("yap")
418 os
.system("git format-patch -k --stdout '%s' > %s" % (commit
, tmpfile
))
419 if subcmd
== "amend":
420 self
.cmd_point(commit
, **{'-f': True})
421 run_command("git stash apply --index %s" % stash
[0])
422 # XXX: use cmd_commit instead
423 os
.system("git commit --amend")
424 stash
= get_output("git stash create")
425 os
.system("git reset --hard")
427 self
.cmd_point("%s^" % commit
, **{'-f': True})
429 stat
= os
.stat(tmpfile
)
432 rc
= os
.system("git am -3 '%s' > /dev/null" % tmpfile
)
434 raise YapError("Failed to apply changes")
436 if subcmd
== "amend" and stash
:
437 run_command("git stash apply %s" % stash
[0])
443 print >> sys
.stderr
, "usage: %s <command>" % sys
.argv
[0]
444 print >> sys
.stderr
, " valid commands: init add rm stage unstage status revert commit uncommit log diff branch switch point history version"
446 def main(self
, args
):
454 debug
= os
.getenv('YAP_DEBUG')
457 meth
= self
.__getattribute
__("cmd_"+command
)
459 if "options" in meth
.__dict
__:
460 flags
, args
= getopt
.getopt(args
, meth
.options
)
466 except (TypeError, getopt
.GetoptError
):
469 print "%s %s %s" % (sys
.argv
[0], command
, meth
.__doc
__)
471 print >> sys
.stderr
, e
473 except AttributeError: