-
Notifications
You must be signed in to change notification settings - Fork 240
/
git-flatten.rb
executable file
·331 lines (306 loc) · 7.64 KB
/
git-flatten.rb
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
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
#!/usr/bin/env ruby
require 'fileutils'
require 'open3'
require 'ftools'
require 'set'
$name = "git flatten"
def usage retcode=0
puts <<USAGE
usage: #{$name} [options] <ref>
USAGE
exit retcode
end
def die msg, usage=false
$stderr.puts msg
usage ? usage(-1) : exit(-1)
end
def execute cmd, opts={}
verbose = opts[:verbose]
debug = opts[:debug]
result = opts[:return] || :to_s
select = opts[:select]
filter = opts[:filter]
res = []
puts cmd if debug
IO.popen cmd do |f|
f.each_line do |line|
l = line.sub(/\n$/,'')
puts l if debug
if !select || select[l]
l = filter ? filter[l] : l
puts l if verbose
res << l
end
end
end
if result == :result
$?.success?
else
if opts[:one]
res.empty? ? nil : res.first
else
res
end
end
end
def git_dir
$git_dir ||= execute("git rev-parse --git-dir", :one => true)
end
def git_editor
$git_editor ||= execute("git config core.editor", :one => true) ||
ENV['VISUAL'] || ENV['EDITOR'] || 'vi'
end
def git_branch
$git_branch ||= execute("git branch",
:select => lambda{|l| l =~ /^\*/},
:filter => lambda{|l| l.sub(/\* /, '')},
:one => true)
end
def ref_to_hash ref
execute("git rev-parse #{ref}", :one => true)
end
def hash_to_str hash
token = '--token--'
execute("git log -1 --pretty=format:'%h#{token}%s' #{hash}",
:one => true).split(/#{token}/)
end
def rev_list *ref
execute("git rev-list --reverse #{ref.join(' ')}")
end
def patch_id hash
execute("git show #{hash} | git patch-id", :one => true)
end
def merge? hash
patch_id(hash).nil?
end
def merge_base c1, c2
execute("git merge-base #{c1} #{c2}", :one => true)
end
def container_branches hash
execute("git branch -a --contains #{hash}",
:filter => lambda{|b| b.sub(/^\s+/,'')})
end
def parent_branches hash
parents = execute("git log -1 --pretty=format:%P #{hash}",
:one => true).split(/\s+/)
branches = Set.new
parents.each do |parent|
branches |= container_branches(parent)
end
branches
end
def parse_flatten target, opts={}
name = "flattenorig"
dirname = "#{git_dir}/refs/#{name}"
filename = "#{dirname}/#{git_branch}"
if Dir[dirname].empty?
Dir.mkdir dirname
end
if Dir[filename].empty? && !opts[:write]
File.open(filename, "w+") do |file|
file.puts merge_base('HEAD', target)
end
end
if opts[:read]
File.open(filename) do |file|
file.readline.sub(/\n$/,'')
end
elsif opts[:write]
File.open(filename,"w") do |file|
file.puts opts[:write]
end
else
filename
end
end
def interactive_edit ref, revs, squash, opts={}
name = "FLATTEN_MSG"
filename = "#{git_dir}/#{name}"
limit = opts[:limit] || 70
if opts[:delete]
File.safe_unlink filename
else
if !opts[:continue]
return [] if revs.empty?
unless File.exists? filename
File.open(filename, "w+") do |file|
str_HEAD = hash_to_str(ref)[0]
str_revs = revs.map do |rev|
hash, str = hash_to_str(rev)
str = str[0..(limit-3)] + '...' if str.length > limit
[hash, str]
end
str_revs.each do |hash, str|
if merge? hash
if (parent_branches(hash) & squash).empty?
action = :squash
else
# merge from a squashed branch
# so it may be picked
action = :pick
end
else
action = :pick
end
res = "%-6s %s %s" % [action, hash, str]
file.puts res
end
file.puts <<USAGE
# Flatten #{str_HEAD}..#{str_revs.last[0]} onto #{str_HEAD}
#
# Commands:
# p, pick = use commit
# e, edit = use commit, but stop for amending
# s, squash = use commit, but meld into previous commit
#
# If you remove a line here THAT COMMIT WILL BE LOST.
# However, if you remove everything, the rebase will be aborted.
#
USAGE
end
end
end
system git_editor, filename if opts[:interactive]
res = []
File.open(filename) do |file|
file.each_line do |line|
unless line =~ /^#/
line = line.sub(/\n$/,'').split(/\s+/)
res << [ line[0], line[1], line[2..-1].join(' ') ]
end
end
end
res
end
end
def store_temp last=nil, opts={}
name = "FLATTEN_TMP"
filename = "#{git_dir}/#{name}"
if opts[:delete]
File.safe_unlink filename
elsif last
File.open(filename, "w+") do |file|
file.puts last
end
else
if File.exists? filename
File.open(filename) do |file|
file.gets.sub(/\n$/,'')
end
else
nil
end
end
end
def parse_action action
case action
when /^e(dit)?$/i
:edit
when /^p(ick)$/i
:pick
when /^s(quash)$/i
:squash
else
die "unknown action '#{action}'"
end
end
def flatten ref, refs, opts={}
if !opts[:abort]
resolv_msg = <<RESOLV
When you have resolved this problem run "git flatten --continue".
If you would prefer to skip this patch, instead run "git flatten --skip".
To restore the original branch and stop flatten run "git flatten --abort".
RESOLV
stored_last = store_temp
if opts[:continue] || opts[:skip]
die "No flatten in progress?" unless stored_last
opts[:interactive] = true
else
die "A flatten is in progress, try --continue, --skip or --abort." if stored_last
squash = opts[:squash] || []
orig = parse_flatten ref, :read => true
target = rev_list ref,
"^#{git_branch}", "^#{orig}",
*[refs, squash.map{|s| "^#{s}"}].flatten
end
last = stored_last
revs = interactive_edit orig, target, squash,
:interactive => opts[:interactive],
:continue => opts[:continue]||opts[:skip]
revs.each do |action, abbrev, str|
hash = ref_to_hash abbrev
if stored_last
if hash != stored_last
puts "Skipping #{abbrev} #{str}"
next
end
end
action = parse_action action
store_temp hash
if action == :squash
puts "Squashing #{abbrev} #{str}"
parse_flatten ref, :write => hash
next
end
if !stored_last
result = execute( "git merge -q --squash #{hash}",
:verbose => true,
:return => :result )
die resolv_msg unless result
die "ok, you can edit it" if action == :edit
end
if !stored_last || !opts[:skip]
result = execute( "git commit -C #{hash}",
:verbose => true,
:return => :result)
die resolv_msg unless result
else
puts "Skipping #{abbrev} #{str}"
end
stored_last = nil
parse_flatten ref, :write => hash
end
if last
parse_flatten ref, :write => last
else
puts "nothing to do"
end
end
store_temp nil, :delete => true
interactive_edit nil, nil, nil, :delete => true
end
ref = nil
refs = []
opts = {}
args = ARGV.dup
while arg = args.shift
case arg
when /--abort/
opts[:abort] = true
ref = :osef
when /--continue/
opts[:continue] = true
ref = :osef
when /-h|--help/
usage
when /-i|--interactive/
opts[:interactive] = true
when /--skip/
opts[:skip] = true
ref = :osef
when /-s|--squash/
opts[:squash] ||= []
opts[:squash] << (arg =~ /=/ ? arg.sub(/.*=\s*/,'') : args.shift )
when /^-/
die "unknow option #{arg}", true
else
case
when ref
refs << arg
else
ref = arg
end
end
end
usage -1 unless ref
flatten ref, refs, opts