-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhistory.py
244 lines (197 loc) · 7.44 KB
/
history.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
import functools
import re
import sublime
from . import GitTextCommand, GitWindowCommand, plugin_file
class GitLog(object):
def run(self, edit=None):
fn = self.get_file_name()
return self.run_log(fn != "", "--", fn)
def run_log(self, follow, *args):
# the ASCII bell (\a) is just a convenient character I'm pretty sure
# won't ever come up in the subject of the commit (and if it does then
# you positively deserve broken output...)
# 9000 is a pretty arbitrarily chosen limit; picked entirely because
# it's about the size of the largest repo I've tested this on... and
# there's a definite hiccup when it's loading that
command = [
"git",
"log",
"--no-color",
"--pretty=%s (%h)\a%an <%aE>\a%ad (%ar)",
"--date=local",
"--max-count=9000",
"--follow" if follow else None,
]
command.extend(args)
self.run_command(command, self.log_done)
def log_done(self, result):
self.results = [r.split("\a", 2) for r in result.strip().split("\n")]
self.quick_panel(self.results, self.log_panel_done)
def log_panel_done(self, picked):
if 0 > picked < len(self.results):
return
item = self.results[picked]
# the commit hash is the last thing on the first line, in brackets
ref = item[0].split(" ")[-1].strip("()")
self.log_result(ref)
def log_result(self, ref):
# I'm not certain I should have the file name here; it restricts the
# details to just the current file. Depends on what the user expects...
# which I'm not sure of.
self.run_command(
["git", "log", "--no-color", "-p", "-1", ref, "--", self.get_file_name()],
self.details_done,
)
def details_done(self, result):
self.scratch(
result,
title="Git Commit Details",
syntax="Packages/Git Formats/Git Log.sublime-syntax",
)
class GitLogCommand(GitLog, GitTextCommand):
pass
class GitLogAllCommand(GitLog, GitWindowCommand):
pass
class GitShow(object):
def run(self, edit=None):
# GitLog Copy-Past
self.run_command(
[
"git",
"log",
"--no-color",
"--pretty=%s (%h)\a%an <%aE>\a%ad (%ar)",
"--date=local",
"--max-count=9000",
"--",
self.get_file_name(),
],
self.show_done,
)
def show_done(self, result):
# GitLog Copy-Past
self.results = [r.split("\a", 2) for r in result.strip().split("\n")]
self.quick_panel(self.results, self.panel_done)
def panel_done(self, picked):
if 0 > picked < len(self.results):
return
item = self.results[picked]
# the commit hash is the last thing on the first line, in brackets
ref = item[0].split(" ")[-1].strip("()")
self.run_command(
["git", "show", "%s:%s" % (ref, self.get_relative_file_path())],
self.details_done,
ref=ref,
)
def details_done(self, result, ref):
syntax = self.view.settings().get("syntax")
self.scratch(result, title="%s:%s" % (ref, self.get_file_name()), syntax=syntax)
class GitShowCommand(GitShow, GitTextCommand):
pass
class GitShowAllCommand(GitShow, GitWindowCommand):
pass
class GitShowCommitCommand(GitWindowCommand):
def run(self, edit=None):
self.window.show_input_panel("Commit to show:", "", self.input_done, None, None)
def input_done(self, commit):
commit = commit.strip()
self.run_command(["git", "show", commit, "--"], self.show_done, commit=commit)
def show_done(self, result, commit):
if result.startswith("fatal:"):
self.panel(result)
return
self.scratch(
result,
title="Git Commit: %s" % commit,
syntax="Packages/Git Formats/Git Log.sublime-syntax",
)
class GitGraph(object):
def run(self, edit=None):
filename = self.get_file_name()
self.run_command(
[
"git",
"log",
"--graph",
"--pretty=%h -%d (%cr) (%ci) <%an> %s",
"--abbrev-commit",
"--no-color",
"--decorate",
"--date=relative",
"--follow" if filename else None,
"--",
filename,
],
self.log_done,
)
def log_done(self, result):
self.scratch(
result,
title="Git Log Graph",
syntax=plugin_file("syntax/Git Graph.tmLanguage"),
)
class GitOpenFileCommand(GitLog, GitWindowCommand):
def run(self):
self.run_command(["git", "branch", "-a", "--no-color"], self.branch_done)
def branch_done(self, result):
self.results = result.rstrip().split("\n")
self.quick_panel(self.results, self.branch_panel_done, sublime.MONOSPACE_FONT)
def branch_panel_done(self, picked):
if 0 > picked < len(self.results):
return
self.branch = self.results[picked].split(" ")[-1]
self.run_log(False, self.branch)
def log_result(self, result_hash):
self.ref = result_hash
self.run_command(
["git", "ls-tree", "-r", "--full-tree", self.ref], self.ls_done
)
def ls_done(self, result):
# Last two items are the ref and the file name
# p.s. has to be a list of lists; tuples cause errors later
self.results = [
[match.group(2), match.group(1)]
for match in re.finditer(r"\S+\s(\S+)\t(.+)", result)
]
self.quick_panel(self.results, self.ls_panel_done)
def ls_panel_done(self, picked):
if 0 > picked < len(self.results):
return
item = self.results[picked]
self.filename = item[0]
self.fileRef = item[1]
self.run_command(["git", "show", self.fileRef], self.show_done)
def show_done(self, result):
self.scratch(result, title="%s:%s" % (self.fileRef, self.filename))
class GitGotoCommit(GitTextCommand):
def run(self, edit):
view = self.view
# Sublime is missing a "find scope in region" API, so we piece one together here:
lines = [view.line(sel.a) for sel in view.sel()]
hashes = self.view.find_by_selector("string.sha")
commits = []
for region in hashes:
for line in lines:
if line.contains(region):
commit = view.substr(region)
if commit.strip("0"):
commits.append(commit)
break
working_dir = view.settings().get("git_root_dir")
for commit in commits:
self.run_command(
["git", "show", commit], self.show_done, working_dir=working_dir
)
def show_done(self, result):
self.scratch(
result,
title="Git Commit View",
syntax="Packages/Git Formats/Git Log.sublime-syntax",
)
def is_enabled(self):
if self.view.element() is not None:
return False
selection = self.view.sel()[0]
return self.view.match_selector(
selection.a, "text.git-blame"
) or self.view.match_selector(selection.a, "text.git-graph")