git-smart-ruby-3 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/Gemfile +10 -0
- data/Gemfile.lock +36 -0
- data/LICENSE.txt +20 -0
- data/README.md +68 -0
- data/Rakefile +71 -0
- data/VERSION +1 -0
- data/bin/git-smart-log +7 -0
- data/bin/git-smart-merge +7 -0
- data/bin/git-smart-pull +7 -0
- data/docs/images/git-smart.png +0 -0
- data/docs/images/smart-log-comparison.png +0 -0
- data/docs/images/smart-pull-screenshot.png +0 -0
- data/docs/introductory_blog_post.md +43 -0
- data/docs/smart-log.html +59 -0
- data/docs/smart-merge.html +212 -0
- data/docs/smart-pull.html +304 -0
- data/git-smart-ruby-3.gemspec +25 -0
- data/lib/commands/smart-log.rb +9 -0
- data/lib/commands/smart-merge.rb +63 -0
- data/lib/commands/smart-pull.rb +117 -0
- data/lib/core_ext/array.rb +9 -0
- data/lib/core_ext/hash.rb +21 -0
- data/lib/core_ext/object.rb +8 -0
- data/lib/git-smart/exceptions.rb +10 -0
- data/lib/git-smart/execution_context.rb +39 -0
- data/lib/git-smart/git_repo.rb +172 -0
- data/lib/git-smart/git_smart.rb +28 -0
- data/lib/git-smart/safe_shell.rb +21 -0
- data/lib/git-smart-ruby-3.rb +8 -0
- data/spec/smart-merge_failures_spec.rb +49 -0
- data/spec/smart-merge_spec.rb +104 -0
- data/spec/smart-pull_spec.rb +215 -0
- data/spec/spec_helper.rb +65 -0
- metadata +141 -0
@@ -0,0 +1,304 @@
|
|
1
|
+
<!DOCTYPE html>
|
2
|
+
<html>
|
3
|
+
<head>
|
4
|
+
<meta http-equiv="content-type" content="text/html;charset=utf-8">
|
5
|
+
<title>smart-pull.rb</title>
|
6
|
+
<link rel="stylesheet" href="http://jashkenas.github.com/docco/resources/docco.css">
|
7
|
+
</head>
|
8
|
+
<body>
|
9
|
+
<div id='container'>
|
10
|
+
<div id="background"></div>
|
11
|
+
<div id="jump_to">
|
12
|
+
Jump To …
|
13
|
+
<div id="jump_wrapper">
|
14
|
+
<div id="jump_page">
|
15
|
+
<a class="source" href="smart-log.html">smart-log.rb</a>
|
16
|
+
<a class="source" href="smart-merge.html">smart-merge.rb</a>
|
17
|
+
<a class="source" href="smart-pull.html">smart-pull.rb</a>
|
18
|
+
</div>
|
19
|
+
</div>
|
20
|
+
</div>
|
21
|
+
<table cellspacing=0 cellpadding=0>
|
22
|
+
<thead>
|
23
|
+
<tr>
|
24
|
+
<th class=docs><h1>smart-pull.rb</h1></th>
|
25
|
+
<th class=code></th>
|
26
|
+
</tr>
|
27
|
+
</thead>
|
28
|
+
<tbody>
|
29
|
+
<tr id='section-1'>
|
30
|
+
<td class=docs>
|
31
|
+
<div class="octowrap">
|
32
|
+
<a class="octothorpe" href="#section-1">#</a>
|
33
|
+
</div>
|
34
|
+
<p>Calling <code>git smart-pull</code> will fetch remote tracked changes
|
35
|
+
and reapply your work on top of it. It’s like a much, much
|
36
|
+
smarter version of <code>git pull --rebase</code>.</p>
|
37
|
+
|
38
|
+
<p>For some background as to why this is needed, see <a href="http://notes.envato.com/developers/rebasing-merge-commits-in-git/">my blog
|
39
|
+
post about the perils of rebasing merge commits</a></p>
|
40
|
+
|
41
|
+
<p>This is how it works:</p>
|
42
|
+
</td>
|
43
|
+
<td class=code>
|
44
|
+
<div class='highlight'><pre><span class="no">GitSmart</span><span class="o">.</span><span class="n">register</span> <span class="s1">'smart-pull'</span> <span class="k">do</span> <span class="o">|</span><span class="n">repo</span><span class="p">,</span> <span class="n">args</span><span class="o">|</span></pre></div>
|
45
|
+
</td>
|
46
|
+
</tr>
|
47
|
+
<tr id='section-2'>
|
48
|
+
<td class=docs>
|
49
|
+
<div class="octowrap">
|
50
|
+
<a class="octothorpe" href="#section-2">#</a>
|
51
|
+
</div>
|
52
|
+
<p>Let’s begin!</p>
|
53
|
+
</td>
|
54
|
+
<td class=code>
|
55
|
+
<div class='highlight'><pre> <span class="n">branch</span> <span class="o">=</span> <span class="n">repo</span><span class="o">.</span><span class="n">current_branch</span>
|
56
|
+
<span class="n">start</span> <span class="s2">"Starting: smart-pull on branch '</span><span class="si">#{</span><span class="n">branch</span><span class="si">}</span><span class="s2">'"</span></pre></div>
|
57
|
+
</td>
|
58
|
+
</tr>
|
59
|
+
<tr id='section-3'>
|
60
|
+
<td class=docs>
|
61
|
+
<div class="octowrap">
|
62
|
+
<a class="octothorpe" href="#section-3">#</a>
|
63
|
+
</div>
|
64
|
+
<p>Let’s not have any arguments, fellas.</p>
|
65
|
+
</td>
|
66
|
+
<td class=code>
|
67
|
+
<div class='highlight'><pre> <span class="nb">warn</span> <span class="s2">"Ignoring arguments: </span><span class="si">#{</span><span class="n">args</span><span class="o">.</span><span class="n">inspect</span><span class="si">}</span><span class="s2">"</span> <span class="k">if</span> <span class="o">!</span><span class="n">args</span><span class="o">.</span><span class="n">empty?</span></pre></div>
|
68
|
+
</td>
|
69
|
+
</tr>
|
70
|
+
<tr id='section-4'>
|
71
|
+
<td class=docs>
|
72
|
+
<div class="octowrap">
|
73
|
+
<a class="octothorpe" href="#section-4">#</a>
|
74
|
+
</div>
|
75
|
+
<p>Try grabbing the tracking remote from the config. If it doesn’t exist,
|
76
|
+
notify the user and default to ‘origin’</p>
|
77
|
+
</td>
|
78
|
+
<td class=code>
|
79
|
+
<div class='highlight'><pre> <span class="n">tracking_remote</span> <span class="o">=</span> <span class="n">repo</span><span class="o">.</span><span class="n">tracking_remote</span> <span class="o">||</span>
|
80
|
+
<span class="n">note</span><span class="p">(</span><span class="s2">"No tracking remote configured, assuming 'origin'"</span><span class="p">)</span> <span class="o">||</span>
|
81
|
+
<span class="s1">'origin'</span></pre></div>
|
82
|
+
</td>
|
83
|
+
</tr>
|
84
|
+
<tr id='section-5'>
|
85
|
+
<td class=docs>
|
86
|
+
<div class="octowrap">
|
87
|
+
<a class="octothorpe" href="#section-5">#</a>
|
88
|
+
</div>
|
89
|
+
<p> Fetch the remote. This pulls down all new commits from the server, not just our branch,
|
90
|
+
but generally that’s a good thing. This is the only communication we need to do with the server.</p>
|
91
|
+
</td>
|
92
|
+
<td class=code>
|
93
|
+
<div class='highlight'><pre> <span class="n">repo</span><span class="o">.</span><span class="n">fetch!</span><span class="p">(</span><span class="n">tracking_remote</span><span class="p">)</span></pre></div>
|
94
|
+
</td>
|
95
|
+
</tr>
|
96
|
+
<tr id='section-6'>
|
97
|
+
<td class=docs>
|
98
|
+
<div class="octowrap">
|
99
|
+
<a class="octothorpe" href="#section-6">#</a>
|
100
|
+
</div>
|
101
|
+
<p>Try grabbing the tracking branch from the config. If it doesn’t exist,
|
102
|
+
notify the user and choose the branch of the same name</p>
|
103
|
+
</td>
|
104
|
+
<td class=code>
|
105
|
+
<div class='highlight'><pre> <span class="n">tracking_branch</span> <span class="o">=</span> <span class="n">repo</span><span class="o">.</span><span class="n">tracking_branch</span> <span class="o">||</span>
|
106
|
+
<span class="n">note</span><span class="p">(</span><span class="s2">"No tracking branch configured, assuming '</span><span class="si">#{</span><span class="n">branch</span><span class="si">}</span><span class="s2">'"</span><span class="p">)</span> <span class="o">||</span>
|
107
|
+
<span class="n">branch</span></pre></div>
|
108
|
+
</td>
|
109
|
+
</tr>
|
110
|
+
<tr id='section-7'>
|
111
|
+
<td class=docs>
|
112
|
+
<div class="octowrap">
|
113
|
+
<a class="octothorpe" href="#section-7">#</a>
|
114
|
+
</div>
|
115
|
+
<p>Check the specified upstream branch exists. Fail if it doesn’t.</p>
|
116
|
+
</td>
|
117
|
+
<td class=code>
|
118
|
+
<div class='highlight'><pre> <span class="n">upstream_branch</span> <span class="o">=</span> <span class="s2">"</span><span class="si">#{</span><span class="n">tracking_remote</span><span class="si">}</span><span class="s2">/</span><span class="si">#{</span><span class="n">tracking_branch</span><span class="si">}</span><span class="s2">"</span>
|
119
|
+
<span class="n">failure</span><span class="p">(</span><span class="s2">"Upstream branch '</span><span class="si">#{</span><span class="n">upstream_branch</span><span class="si">}</span><span class="s2">' doesn't exist!"</span><span class="p">)</span> <span class="k">if</span> <span class="o">!</span><span class="n">repo</span><span class="o">.</span><span class="n">exists?</span><span class="p">(</span><span class="n">upstream_branch</span><span class="p">)</span></pre></div>
|
120
|
+
</td>
|
121
|
+
</tr>
|
122
|
+
<tr id='section-8'>
|
123
|
+
<td class=docs>
|
124
|
+
<div class="octowrap">
|
125
|
+
<a class="octothorpe" href="#section-8">#</a>
|
126
|
+
</div>
|
127
|
+
<p>Grab the SHAs of the commits we’ll be working with.</p>
|
128
|
+
</td>
|
129
|
+
<td class=code>
|
130
|
+
<div class='highlight'><pre> <span class="n">head</span> <span class="o">=</span> <span class="n">repo</span><span class="o">.</span><span class="n">sha</span><span class="p">(</span><span class="s1">'HEAD'</span><span class="p">)</span>
|
131
|
+
<span class="n">remote</span> <span class="o">=</span> <span class="n">repo</span><span class="o">.</span><span class="n">sha</span><span class="p">(</span><span class="n">upstream_branch</span><span class="p">)</span></pre></div>
|
132
|
+
</td>
|
133
|
+
</tr>
|
134
|
+
<tr id='section-9'>
|
135
|
+
<td class=docs>
|
136
|
+
<div class="octowrap">
|
137
|
+
<a class="octothorpe" href="#section-9">#</a>
|
138
|
+
</div>
|
139
|
+
<p>If both HEAD and our upstream_branch resolve to the same SHA, there’s nothing to do!</p>
|
140
|
+
</td>
|
141
|
+
<td class=code>
|
142
|
+
<div class='highlight'><pre> <span class="k">if</span> <span class="n">head</span> <span class="o">==</span> <span class="n">remote</span>
|
143
|
+
<span class="nb">puts</span> <span class="s2">"Neither your local branch '</span><span class="si">#{</span><span class="n">branch</span><span class="si">}</span><span class="s2">', nor the remote branch '</span><span class="si">#{</span><span class="n">upstream_branch</span><span class="si">}</span><span class="s2">' have moved on."</span>
|
144
|
+
<span class="n">success</span> <span class="s2">"Already up-to-date"</span>
|
145
|
+
<span class="k">else</span></pre></div>
|
146
|
+
</td>
|
147
|
+
</tr>
|
148
|
+
<tr id='section-10'>
|
149
|
+
<td class=docs>
|
150
|
+
<div class="octowrap">
|
151
|
+
<a class="octothorpe" href="#section-10">#</a>
|
152
|
+
</div>
|
153
|
+
<p>Find out where the two branches diverged using merge-base. It’s what git
|
154
|
+
uses internally.</p>
|
155
|
+
</td>
|
156
|
+
<td class=code>
|
157
|
+
<div class='highlight'><pre> <span class="n">merge_base</span> <span class="o">=</span> <span class="n">repo</span><span class="o">.</span><span class="n">merge_base</span><span class="p">(</span><span class="n">head</span><span class="p">,</span> <span class="n">remote</span><span class="p">)</span></pre></div>
|
158
|
+
</td>
|
159
|
+
</tr>
|
160
|
+
<tr id='section-11'>
|
161
|
+
<td class=docs>
|
162
|
+
<div class="octowrap">
|
163
|
+
<a class="octothorpe" href="#section-11">#</a>
|
164
|
+
</div>
|
165
|
+
<p>Report how many commits are new locally, since that’s useful information.</p>
|
166
|
+
</td>
|
167
|
+
<td class=code>
|
168
|
+
<div class='highlight'><pre> <span class="n">new_commits_locally</span> <span class="o">=</span> <span class="n">repo</span><span class="o">.</span><span class="n">rev_list</span><span class="p">(</span><span class="n">merge_base</span><span class="p">,</span> <span class="n">head</span><span class="p">)</span>
|
169
|
+
<span class="k">if</span> <span class="o">!</span><span class="n">new_commits_locally</span><span class="o">.</span><span class="n">empty?</span>
|
170
|
+
<span class="n">note</span> <span class="s2">"You have </span><span class="si">#{</span><span class="n">new_commits_locally</span><span class="o">.</span><span class="n">length</span><span class="si">}</span><span class="s2"> new commit</span><span class="si">#{</span><span class="s1">'s'</span> <span class="k">if</span> <span class="n">new_commits_locally</span><span class="o">.</span><span class="n">length</span> <span class="o">!=</span> <span class="mi">1</span><span class="si">}</span><span class="s2"> on '</span><span class="si">#{</span><span class="n">branch</span><span class="si">}</span><span class="s2">'."</span>
|
171
|
+
<span class="k">end</span></pre></div>
|
172
|
+
</td>
|
173
|
+
</tr>
|
174
|
+
<tr id='section-12'>
|
175
|
+
<td class=docs>
|
176
|
+
<div class="octowrap">
|
177
|
+
<a class="octothorpe" href="#section-12">#</a>
|
178
|
+
</div>
|
179
|
+
<p>By comparing the merge_base to both HEAD and the remote, we can
|
180
|
+
determine whether both or only one have moved on.
|
181
|
+
If the remote hasn’t changed, we’re already up to date, so there’s nothing
|
182
|
+
to pull.</p>
|
183
|
+
</td>
|
184
|
+
<td class=code>
|
185
|
+
<div class='highlight'><pre> <span class="k">if</span> <span class="n">merge_base</span> <span class="o">==</span> <span class="n">remote</span>
|
186
|
+
<span class="nb">puts</span> <span class="s2">"Remote branch '</span><span class="si">#{</span><span class="n">upstream_branch</span><span class="si">}</span><span class="s2">' has not moved on."</span>
|
187
|
+
<span class="n">success</span> <span class="s2">"Already up-to-date"</span>
|
188
|
+
<span class="k">else</span></pre></div>
|
189
|
+
</td>
|
190
|
+
</tr>
|
191
|
+
<tr id='section-13'>
|
192
|
+
<td class=docs>
|
193
|
+
<div class="octowrap">
|
194
|
+
<a class="octothorpe" href="#section-13">#</a>
|
195
|
+
</div>
|
196
|
+
<p>If the remote <em>has</em> moved on, we actually have some work to do:</p>
|
197
|
+
|
198
|
+
<p>First, report how many commits are new on remote. Because that’s useful information, too.</p>
|
199
|
+
</td>
|
200
|
+
<td class=code>
|
201
|
+
<div class='highlight'><pre> <span class="n">new_commits_on_remote</span> <span class="o">=</span> <span class="n">repo</span><span class="o">.</span><span class="n">rev_list</span><span class="p">(</span><span class="n">merge_base</span><span class="p">,</span> <span class="n">remote</span><span class="p">)</span>
|
202
|
+
<span class="n">is_are</span><span class="p">,</span> <span class="n">s_or_not</span> <span class="o">=</span> <span class="p">(</span><span class="n">new_commits_on_remote</span><span class="o">.</span><span class="n">length</span> <span class="o">==</span> <span class="mi">1</span><span class="p">)</span> <span class="p">?</span> <span class="o">[</span><span class="s1">'is'</span><span class="p">,</span> <span class="s1">''</span><span class="o">]</span> <span class="p">:</span> <span class="o">[</span><span class="s1">'are'</span><span class="p">,</span> <span class="s1">'s'</span><span class="o">]</span>
|
203
|
+
<span class="n">note</span> <span class="s2">"There </span><span class="si">#{</span><span class="n">is_are</span><span class="si">}</span><span class="s2"> </span><span class="si">#{</span><span class="n">new_commits_on_remote</span><span class="o">.</span><span class="n">length</span><span class="si">}</span><span class="s2"> new commit</span><span class="si">#{</span><span class="n">s_or_not</span><span class="si">}</span><span class="s2"> on '</span><span class="si">#{</span><span class="n">upstream_branch</span><span class="si">}</span><span class="s2">'."</span></pre></div>
|
204
|
+
</td>
|
205
|
+
</tr>
|
206
|
+
<tr id='section-14'>
|
207
|
+
<td class=docs>
|
208
|
+
<div class="octowrap">
|
209
|
+
<a class="octothorpe" href="#section-14">#</a>
|
210
|
+
</div>
|
211
|
+
<p>Next, detect if there are local changes and stash them.</p>
|
212
|
+
</td>
|
213
|
+
<td class=code>
|
214
|
+
<div class='highlight'><pre> <span class="n">stash_required</span> <span class="o">=</span> <span class="n">repo</span><span class="o">.</span><span class="n">dirty?</span>
|
215
|
+
<span class="k">if</span> <span class="n">stash_required</span>
|
216
|
+
<span class="n">note</span> <span class="s2">"Working directory dirty. Stashing..."</span>
|
217
|
+
<span class="n">repo</span><span class="o">.</span><span class="n">stash!</span>
|
218
|
+
<span class="k">end</span>
|
219
|
+
|
220
|
+
<span class="n">success_messages</span> <span class="o">=</span> <span class="o">[]</span></pre></div>
|
221
|
+
</td>
|
222
|
+
</tr>
|
223
|
+
<tr id='section-15'>
|
224
|
+
<td class=docs>
|
225
|
+
<div class="octowrap">
|
226
|
+
<a class="octothorpe" href="#section-15">#</a>
|
227
|
+
</div>
|
228
|
+
<p>Then, bring the local branch up to date.</p>
|
229
|
+
|
230
|
+
<p>If our local branch hasn’t moved on, that’s easy – we just need to fast-forward.</p>
|
231
|
+
</td>
|
232
|
+
<td class=code>
|
233
|
+
<div class='highlight'><pre> <span class="k">if</span> <span class="n">merge_base</span> <span class="o">==</span> <span class="n">head</span>
|
234
|
+
<span class="nb">puts</span> <span class="s2">"Local branch '</span><span class="si">#{</span><span class="n">branch</span><span class="si">}</span><span class="s2">' has not moved on. Fast-forwarding..."</span>
|
235
|
+
<span class="n">repo</span><span class="o">.</span><span class="n">fast_forward!</span><span class="p">(</span><span class="n">upstream_branch</span><span class="p">)</span>
|
236
|
+
<span class="n">success_messages</span> <span class="o"><<</span> <span class="s2">"Fast forwarded from </span><span class="si">#{</span><span class="n">head</span><span class="o">[</span><span class="mi">0</span><span class="p">,</span><span class="mi">7</span><span class="o">]</span><span class="si">}</span><span class="s2"> to </span><span class="si">#{</span><span class="n">remote</span><span class="o">[</span><span class="mi">0</span><span class="p">,</span><span class="mi">7</span><span class="o">]</span><span class="si">}</span><span class="s2">"</span>
|
237
|
+
<span class="k">else</span></pre></div>
|
238
|
+
</td>
|
239
|
+
</tr>
|
240
|
+
<tr id='section-16'>
|
241
|
+
<td class=docs>
|
242
|
+
<div class="octowrap">
|
243
|
+
<a class="octothorpe" href="#section-16">#</a>
|
244
|
+
</div>
|
245
|
+
<p>If our local branch has new commits, we need to rebase them on top of master.</p>
|
246
|
+
|
247
|
+
<p>When we rebase, we use <code>git rebase -p</code>, which attempts to recreate merges
|
248
|
+
instead of ignoring them. For a description as to why, see my <a href="http://notes.envato.com/developers/rebasing-merge-commits-in-git/">blog post</a>.</p>
|
249
|
+
</td>
|
250
|
+
<td class=code>
|
251
|
+
<div class='highlight'><pre> <span class="n">note</span> <span class="s2">"Both local and remote branches have moved on. Branch 'master' needs to be rebased onto 'origin/master'"</span>
|
252
|
+
<span class="n">repo</span><span class="o">.</span><span class="n">rebase_preserving_merges!</span><span class="p">(</span><span class="n">upstream_branch</span><span class="p">)</span>
|
253
|
+
<span class="n">success_messages</span> <span class="o"><<</span> <span class="s2">"HEAD moved from </span><span class="si">#{</span><span class="n">head</span><span class="o">[</span><span class="mi">0</span><span class="p">,</span><span class="mi">7</span><span class="o">]</span><span class="si">}</span><span class="s2"> to </span><span class="si">#{</span><span class="n">repo</span><span class="o">.</span><span class="n">sha</span><span class="p">(</span><span class="s1">'HEAD'</span><span class="p">)</span><span class="o">[</span><span class="mi">0</span><span class="p">,</span><span class="mi">7</span><span class="o">]</span><span class="si">}</span><span class="s2">."</span>
|
254
|
+
<span class="k">end</span></pre></div>
|
255
|
+
</td>
|
256
|
+
</tr>
|
257
|
+
<tr id='section-17'>
|
258
|
+
<td class=docs>
|
259
|
+
<div class="octowrap">
|
260
|
+
<a class="octothorpe" href="#section-17">#</a>
|
261
|
+
</div>
|
262
|
+
<p>If we stashed before, pop now.</p>
|
263
|
+
</td>
|
264
|
+
<td class=code>
|
265
|
+
<div class='highlight'><pre> <span class="k">if</span> <span class="n">stash_required</span>
|
266
|
+
<span class="n">note</span> <span class="s2">"Reapplying local changes..."</span>
|
267
|
+
<span class="n">repo</span><span class="o">.</span><span class="n">stash_pop!</span>
|
268
|
+
<span class="k">end</span></pre></div>
|
269
|
+
</td>
|
270
|
+
</tr>
|
271
|
+
<tr id='section-18'>
|
272
|
+
<td class=docs>
|
273
|
+
<div class="octowrap">
|
274
|
+
<a class="octothorpe" href="#section-18">#</a>
|
275
|
+
</div>
|
276
|
+
<p>Display a nice completion message in large, friendly letters.</p>
|
277
|
+
</td>
|
278
|
+
<td class=code>
|
279
|
+
<div class='highlight'><pre> <span class="n">success</span> <span class="o">[</span><span class="s2">"All good."</span><span class="p">,</span> <span class="o">*</span><span class="n">success_messages</span><span class="o">].</span><span class="n">join</span><span class="p">(</span><span class="s2">" "</span><span class="p">)</span>
|
280
|
+
<span class="k">end</span></pre></div>
|
281
|
+
</td>
|
282
|
+
</tr>
|
283
|
+
<tr id='section-19'>
|
284
|
+
<td class=docs>
|
285
|
+
<div class="octowrap">
|
286
|
+
<a class="octothorpe" href="#section-19">#</a>
|
287
|
+
</div>
|
288
|
+
<p>Still to do:</p>
|
289
|
+
|
290
|
+
<ul>
|
291
|
+
<li>Ensure ORIG_HEAD is correctly set at the end of each run.</li>
|
292
|
+
<li>If the rebase fails, and you’ve done a stash, remind the user to unstash</li>
|
293
|
+
</ul>
|
294
|
+
|
295
|
+
|
296
|
+
</td>
|
297
|
+
<td class=code>
|
298
|
+
<div class='highlight'><pre> <span class="k">end</span>
|
299
|
+
<span class="k">end</span></pre></div>
|
300
|
+
</td>
|
301
|
+
</tr>
|
302
|
+
</table>
|
303
|
+
</div>
|
304
|
+
</body>
|
@@ -0,0 +1,25 @@
|
|
1
|
+
Gem::Specification.new do |s|
|
2
|
+
s.name = %q{git-smart-ruby-3}
|
3
|
+
s.version = "0.0.1"
|
4
|
+
|
5
|
+
s.authors = ["Glen Maddern"]
|
6
|
+
s.email = %q{glenmaddern@gmail.com}
|
7
|
+
s.date = %q{2025-03-14}
|
8
|
+
s.summary = %q{Add some smarts to your git workflow}
|
9
|
+
s.description = %q{Installs additional 'smart' git commands, like `git smart-pull`. Ruby 3 compatible version.}
|
10
|
+
s.homepage = %q{http://github.com/YOUR_USERNAME/git-smart-ruby-3}
|
11
|
+
s.licenses = ["MIT"]
|
12
|
+
s.require_paths = ["lib"]
|
13
|
+
|
14
|
+
s.extra_rdoc_files = %w[LICENSE.txt README.md]
|
15
|
+
|
16
|
+
s.executables = Dir["bin/*"].map { |f| File.basename(f) }
|
17
|
+
s.files = Dir["{lib,docs}/**/*"] + %w[Gemfile Gemfile.lock LICENSE.txt README.md Rakefile VERSION git-smart-ruby-3.gemspec]
|
18
|
+
s.test_files = Dir["spec/**/*"]
|
19
|
+
|
20
|
+
s.add_runtime_dependency(%q<colorize>, [">= 0"])
|
21
|
+
|
22
|
+
s.add_development_dependency(%q<rspec>, [">= 2.7.0"])
|
23
|
+
s.add_development_dependency(%q<rcov>, [">= 0"])
|
24
|
+
s.add_development_dependency(%q<rocco>, [">= 0"])
|
25
|
+
end
|
@@ -0,0 +1,9 @@
|
|
1
|
+
#This is a super simple alias for the most badass of log outputs that git
|
2
|
+
#offers. Uses git log --graph under the hood.
|
3
|
+
#
|
4
|
+
#Thanks to [@ben_h](http://twitter.com/ben_h) for this one!
|
5
|
+
GitSmart.register 'smart-log' do |repo, args|
|
6
|
+
#Super simple, passes the args through to git log, but
|
7
|
+
#ratchets up the badassness quotient.
|
8
|
+
repo.log_to_shell('--pretty=format:%C(yellow)%h%Cblue%d%Creset %s %C(white) %an, %ar%Creset', '--graph', *args)
|
9
|
+
end
|
@@ -0,0 +1,63 @@
|
|
1
|
+
#Calling `git smart-merge branchname` will, quite simply, perform a
|
2
|
+
#non-fast-forward merge wrapped in a stash push/pop, if that's required.
|
3
|
+
#With some helpful extra output.
|
4
|
+
GitSmart.register 'smart-merge' do |repo, args|
|
5
|
+
#Let's begin!
|
6
|
+
current_branch = repo.current_branch
|
7
|
+
start "Starting: smart-merge on branch '#{current_branch}'"
|
8
|
+
|
9
|
+
#Grab the merge_target the user specified
|
10
|
+
merge_target = args.shift
|
11
|
+
failure "Usage: git smart-merge ref" if !merge_target
|
12
|
+
|
13
|
+
#Make sure git can resolve the reference to the merge_target
|
14
|
+
merge_sha = repo.sha(merge_target)
|
15
|
+
failure "Branch to merge '#{merge_target}' not recognised by git!" if !merge_sha
|
16
|
+
|
17
|
+
#If the SHA of HEAD and the merge_target are the same, we're trying to merge
|
18
|
+
#the same commit with itself. Which is madness!
|
19
|
+
head = repo.sha('HEAD')
|
20
|
+
if merge_sha == head
|
21
|
+
note "Branch '#{merge_target}' has no new commits. Nothing to merge in."
|
22
|
+
success 'Already up-to-date.'
|
23
|
+
else
|
24
|
+
#Determine the merge-base of the two commits, so we can report some useful output
|
25
|
+
#about how many new commits have been added.
|
26
|
+
merge_base = repo.merge_base(head, merge_sha)
|
27
|
+
|
28
|
+
#Report the number of commits on merge_target we're about to merge in.
|
29
|
+
new_commits_on_merge_target = repo.rev_list(merge_base, merge_target)
|
30
|
+
puts "Branch '#{merge_target}' has diverged by #{new_commits_on_merge_target.length} commit#{'s' if new_commits_on_merge_target.length != 1}. Merging in."
|
31
|
+
|
32
|
+
#Determine if our branch has moved on.
|
33
|
+
if head == merge_base
|
34
|
+
#Note: Even though we _can_ fast-forward here, it's a really bad idea since
|
35
|
+
#it results in the disappearance of the branch in history. For a good discussion
|
36
|
+
#on this topic, see this [StackOverflow question](http://stackoverflow.com/questions/2850369/why-uses-git-fast-forward-merging-per-default).
|
37
|
+
note "Branch '#{current_branch}' has not moved on since '#{merge_target}' diverged. Running with --no-ff anyway, since a fast-forward is unexpected behaviour."
|
38
|
+
else
|
39
|
+
#Report how many commits on our branch since merge_target diverged.
|
40
|
+
new_commits_on_branch = repo.rev_list(merge_base, head)
|
41
|
+
puts "Branch '#{current_branch}' has #{new_commits_on_branch.length} new commit#{'s' if new_commits_on_merge_target.length != 1} since '#{merge_target}' diverged."
|
42
|
+
end
|
43
|
+
|
44
|
+
#Before we merge, detect if there are local changes and stash them.
|
45
|
+
stash_required = repo.dirty?
|
46
|
+
if stash_required
|
47
|
+
note "Working directory dirty. Stashing..."
|
48
|
+
repo.stash!
|
49
|
+
end
|
50
|
+
|
51
|
+
#Perform the merge, using --no-ff.
|
52
|
+
repo.merge_no_ff!(merge_target)
|
53
|
+
|
54
|
+
#If we stashed before, pop now.
|
55
|
+
if stash_required
|
56
|
+
note "Reapplying local changes..."
|
57
|
+
repo.stash_pop!
|
58
|
+
end
|
59
|
+
|
60
|
+
#Display a nice completion message in large, friendly letters.
|
61
|
+
success "All good. Created merge commit #{repo.sha('HEAD')[0,7]}."
|
62
|
+
end
|
63
|
+
end
|
@@ -0,0 +1,117 @@
|
|
1
|
+
#Calling `git smart-pull` will fetch remote tracked changes
|
2
|
+
#and reapply your work on top of it. It's like a much, much
|
3
|
+
#smarter version of `git pull --rebase`.
|
4
|
+
#
|
5
|
+
#For some background as to why this is needed, see [my blog
|
6
|
+
#post about the perils of rebasing merge commits](http://notes.envato.com/developers/rebasing-merge-commits-in-git/)
|
7
|
+
#
|
8
|
+
#This is how it works:
|
9
|
+
|
10
|
+
GitSmart.register 'smart-pull' do |repo, args|
|
11
|
+
#Let's begin!
|
12
|
+
branch = repo.current_branch
|
13
|
+
start "Starting: smart-pull on branch '#{branch}'"
|
14
|
+
|
15
|
+
#Let's not have any arguments, fellas.
|
16
|
+
warn "Ignoring arguments: #{args.inspect}" if !args.empty?
|
17
|
+
|
18
|
+
#Try grabbing the tracking remote from the config. If it doesn't exist,
|
19
|
+
#notify the user and default to 'origin'
|
20
|
+
tracking_remote = repo.tracking_remote ||
|
21
|
+
note("No tracking remote configured, assuming 'origin'") ||
|
22
|
+
'origin'
|
23
|
+
|
24
|
+
# Fetch the remote. This pulls down all new commits from the server, not just our branch,
|
25
|
+
# but generally that's a good thing. This is the only communication we need to do with the server.
|
26
|
+
repo.fetch!(tracking_remote)
|
27
|
+
|
28
|
+
#Try grabbing the tracking branch from the config. If it doesn't exist,
|
29
|
+
#notify the user and choose the branch of the same name
|
30
|
+
tracking_branch = repo.tracking_branch ||
|
31
|
+
note("No tracking branch configured, assuming '#{branch}'") ||
|
32
|
+
branch
|
33
|
+
|
34
|
+
#Check the specified upstream branch exists. Fail if it doesn't.
|
35
|
+
upstream_branch = "#{tracking_remote}/#{tracking_branch}"
|
36
|
+
failure("Upstream branch '#{upstream_branch}' doesn't exist!") if !repo.exists?(upstream_branch)
|
37
|
+
|
38
|
+
#Grab the SHAs of the commits we'll be working with.
|
39
|
+
head = repo.sha('HEAD')
|
40
|
+
remote = repo.sha(upstream_branch)
|
41
|
+
|
42
|
+
#If both HEAD and our upstream_branch resolve to the same SHA, there's nothing to do!
|
43
|
+
if head == remote
|
44
|
+
puts "Neither your local branch '#{branch}', nor the remote branch '#{upstream_branch}' have moved on."
|
45
|
+
success "Already up-to-date"
|
46
|
+
else
|
47
|
+
#Find out where the two branches diverged using merge-base. It's what git
|
48
|
+
#uses internally.
|
49
|
+
merge_base = repo.merge_base(head, remote)
|
50
|
+
|
51
|
+
#Report how many commits are new locally, since that's useful information.
|
52
|
+
new_commits_locally = repo.rev_list(merge_base, head)
|
53
|
+
if !new_commits_locally.empty?
|
54
|
+
note "You have #{new_commits_locally.length} new commit#{'s' if new_commits_locally.length != 1} on '#{branch}'."
|
55
|
+
end
|
56
|
+
|
57
|
+
#By comparing the merge_base to both HEAD and the remote, we can
|
58
|
+
#determine whether both or only one have moved on.
|
59
|
+
#If the remote hasn't changed, we're already up to date, so there's nothing
|
60
|
+
#to pull.
|
61
|
+
if merge_base == remote
|
62
|
+
puts "Remote branch '#{upstream_branch}' has not moved on."
|
63
|
+
success "Already up-to-date"
|
64
|
+
else
|
65
|
+
#If the remote _has_ moved on, we actually have some work to do:
|
66
|
+
#
|
67
|
+
#First, report how many commits are new on remote. Because that's useful information, too.
|
68
|
+
new_commits_on_remote = repo.rev_list(merge_base, remote)
|
69
|
+
is_are, s_or_not = (new_commits_on_remote.length == 1) ? ['is', ''] : ['are', 's']
|
70
|
+
note "There #{is_are} #{new_commits_on_remote.length} new commit#{s_or_not} on '#{upstream_branch}'."
|
71
|
+
|
72
|
+
#Next, detect if there are local changes and stash them.
|
73
|
+
stash_required = repo.dirty?
|
74
|
+
if stash_required
|
75
|
+
note "Working directory dirty. Stashing..."
|
76
|
+
repo.stash!
|
77
|
+
end
|
78
|
+
|
79
|
+
success_messages = []
|
80
|
+
|
81
|
+
#Then, bring the local branch up to date.
|
82
|
+
#
|
83
|
+
#If our local branch hasn't moved on, that's easy - we just need to fast-forward.
|
84
|
+
if merge_base == head
|
85
|
+
puts "Local branch '#{branch}' has not moved on. Fast-forwarding..."
|
86
|
+
repo.fast_forward!(upstream_branch)
|
87
|
+
success_messages << "Fast forwarded from #{head[0,7]} to #{remote[0,7]}"
|
88
|
+
else
|
89
|
+
#If our local branch has new commits, we need to rebase them on top of master.
|
90
|
+
#
|
91
|
+
#When we rebase, we use `git rebase -p`, which attempts to recreate merges
|
92
|
+
#instead of ignoring them. For a description as to why, see my [blog post](http://notes.envato.com/developers/rebasing-merge-commits-in-git/).
|
93
|
+
note "Both local and remote branches have moved on. Branch 'master' needs to be rebased onto 'origin/master'"
|
94
|
+
repo.rebase_preserving_merges!(upstream_branch)
|
95
|
+
success_messages << "HEAD moved from #{head[0,7]} to #{repo.sha('HEAD')[0,7]}."
|
96
|
+
end
|
97
|
+
|
98
|
+
#If we stashed before, pop now.
|
99
|
+
if stash_required
|
100
|
+
note "Reapplying local changes..."
|
101
|
+
repo.stash_pop!
|
102
|
+
end
|
103
|
+
|
104
|
+
#Use smart-log to show the new commits.
|
105
|
+
GitSmart.run('smart-log', ["#{merge_base}..#{upstream_branch}"])
|
106
|
+
|
107
|
+
#Display a nice completion message in large, friendly letters.
|
108
|
+
success ["All good.", *success_messages].join(" ")
|
109
|
+
end
|
110
|
+
|
111
|
+
#Still to do:
|
112
|
+
#
|
113
|
+
#* Ensure ORIG_HEAD is correctly set at the end of each run.
|
114
|
+
#* If the rebase fails, and you've done a stash, remind the user to unstash
|
115
|
+
|
116
|
+
end
|
117
|
+
end
|
@@ -0,0 +1,21 @@
|
|
1
|
+
class Hash
|
2
|
+
def map_keys &blk
|
3
|
+
map_keys_with_values { |k,v| blk.call(k) }
|
4
|
+
end
|
5
|
+
|
6
|
+
def map_keys_with_values &blk
|
7
|
+
result = {}
|
8
|
+
each { |k,v| result[blk.call(k,v)] = v}
|
9
|
+
result
|
10
|
+
end
|
11
|
+
|
12
|
+
def map_values &blk
|
13
|
+
map_values_with_keys { |k,v| blk.call(v) }
|
14
|
+
end
|
15
|
+
|
16
|
+
def map_values_with_keys &blk
|
17
|
+
result = {}
|
18
|
+
each { |k,v| result[k] = blk.call(k,v)}
|
19
|
+
result
|
20
|
+
end
|
21
|
+
end
|
@@ -0,0 +1,39 @@
|
|
1
|
+
# The context that commands get executed within. Used for defining and scoping helper methods.
|
2
|
+
|
3
|
+
class ExecutionContext
|
4
|
+
def initialize
|
5
|
+
require 'colorize'
|
6
|
+
end
|
7
|
+
|
8
|
+
def start msg
|
9
|
+
puts "- #{msg} -".green
|
10
|
+
end
|
11
|
+
|
12
|
+
def note msg
|
13
|
+
puts "* #{msg}"
|
14
|
+
end
|
15
|
+
|
16
|
+
def warn msg
|
17
|
+
puts msg.red
|
18
|
+
end
|
19
|
+
|
20
|
+
def puts_with_done msg, &blk
|
21
|
+
print "#{msg}..."
|
22
|
+
blk.call
|
23
|
+
puts "done."
|
24
|
+
end
|
25
|
+
|
26
|
+
def success msg
|
27
|
+
puts big_message(msg).green
|
28
|
+
end
|
29
|
+
|
30
|
+
def failure msg
|
31
|
+
puts big_message(msg).red
|
32
|
+
raise GitSmart::RunFailed
|
33
|
+
end
|
34
|
+
|
35
|
+
def big_message msg
|
36
|
+
spacer_line = (" " + "-" * (msg.length + 20) + " ")
|
37
|
+
[spacer_line, "|" + " " * 10 + msg + " " * 10 + "|", spacer_line].join("\n")
|
38
|
+
end
|
39
|
+
end
|