comparison docs/user-guide.rst @ 978:8328337d23b2

docs: add new user guide This has also been reviewed to death on evolve-testers. There are still a couple of short sections to write (clearly marked "TODO"), and one example to add. But (if I may be so bold) this is a gigantic improvement over the current docs, so it really should get merged. Incidentally, the figures are all SVG files created with Inkscape. They're not perfect, but they're pretty nice. Anyone who knows a better way to create technical diagrams is welcome to contribute. One wart: there's a test script that accompanies the document and largely duplicates it, but I haven't unified them. I've been concentrating on writing the best possible content, not on fiddling with tools. I suspect that unifying them will be non-trivial, but definitely worth doing.
author Greg Ward <greg@gerg.ca>
date Thu, 05 Jun 2014 22:11:04 -0400
parents
children 8cc6e90354a9
comparison
equal deleted inserted replaced
977:cc0f0d94bf30 978:8328337d23b2
1 .. Copyright © 2014 Greg Ward <greg@gerg.ca>
2
3 ------------------
4 Evolve: User Guide
5 ------------------
6
7 .. contents::
8
9 Life without ``evolve``
10 -----------------------
11
12 Before we dive into learning about ``evolve``, let's look into some
13 features of core Mercurial that interact with ``evolve``. ``commit``
14 affects ``evolve``, and ``evolve`` modifies how ``commit --amend``
15 works.
16
17 Example 1: Commit a new changeset
18 =================================
19
20 To create a new changeset, simply run ``hg commit`` as usual.
21 ``evolve`` does not change the behaviour of ``commit`` at all.
22
23 However, it's important to understand that new changesets are in the
24 *draft* phase by default: they are mutable. This means that they can
25 be modified by Mercurial's existing history-editing commands
26 (``rebase``, ``histedit``, etc.), and also by the ``evolve``
27 extension. Specifically, ``evolve`` adds a number of commands that can
28 be used to modify history: ``amend``, ``uncommit``, ``prune``,
29 ``fold``, and ``evolve``. Generally speaking, changesets remain in
30 *draft* phase until they are pushed to another repository, at which
31 point they enter *public* phase. ::
32
33 $ hg commit -m 'implement feature X'
34 $ hg phase -r .
35 1: draft
36
37 (Strictly speaking, changesets only become public when they are pushed
38 to a *publishing* repository. But all repositories are publishing by
39 default; you have to explicitly configure repositories to be
40 *non-publishing*. Non-publishing repositories are an advanced topic
41 which we'll see when we get to `sharing mutable history`_.)
42
43 .. _`sharing mutable history`: sharing.html
44
45 Example 2: Amend a changeset (traditional)
46 ==========================================
47
48 Imagine you've just committed a new changeset, and then you discover a
49 mistake. Maybe you forgot to run the tests and a failure slipped in.
50 You want to modify history so that you push one perfect changeset,
51 rather than one flawed changeset followed by an "oops" commit. (Or
52 perhaps you made a typo in the commit message—this is really feature
53 *Y*, not feature X. You can't fix that with a followup commit.)
54
55 This is actually trivial with plain vanilla Mercurial since 2.2: fix
56 your mistake and run ::
57
58 $ hg commit --amend -m 'implement feature Y'
59
60 to create a new, amended changeset. The drawback of doing this with
61 vanilla Mercurial is that your original, flawed, changeset is removed
62 from the repository. This is *unsafe* history editing. It's probably
63 not too serious if all you did was fix a syntax error, but still.
64
65 .. figure:: figures/figure-ug01.svg
66
67 Figure 1: unsafe history modification with core Mercurial (not
68 using ``evolve``): the original revision 1 is destroyed.
69
70 (Incidentally, Mercurial's traditional history modification mechanism
71 isn't *really* unsafe: any changeset(s) removed from the repository
72 are kept in a backup directory, so you can manually restore them later
73 if you change your mind. But it's awkward and inconvenient compared to
74 the features provided by ``evolve`` and changeset obsolescence.)
75
76 Life with ``evolve`` (basic usage)
77 ----------------------------------
78
79 Once you enable the ``evolve`` extension, a number of features are
80 available to you. First, we're going to explore several examples of
81 painless, trouble-free history modification.
82
83 Example 3: Amend a changeset (with ``evolve``)
84 ==============================================
85
86 Outwardly, amending a changeset with ``evolve`` can look exactly the
87 same as it does with core Mercurial (example 2)::
88
89 $ hg commit --amend -m 'implement feature Y'
90
91 Alternately, you can use the new ``amend`` command added by
92 ``evolve``::
93
94 $ hg amend -m 'implement feature Y'
95
96 (``hg amend`` is nearly synonymous with ``hg commit --amend``. The
97 difference is that ``hg amend`` reuses the existing commit message by
98 default, whereas ``hg commit --amend`` runs your editor if you don't
99 pass ``-m`` or ``-l``.)
100
101 Under the hood, though, things are quite different. Mercurial has
102 simply marked the old changeset *obsolete*, replacing it with a new
103 one. We'll explore what this means in detail later, after working
104 through a few more examples.
105
106 Example 4: Prune an unwanted changeset
107 ======================================
108
109 Sometimes you make a change, and then decide it was such a bad idea
110 that you don't want anyone to know about it. Or maybe it was a
111 debugging hack that you needed to keep around for a while, but do not
112 intend to ever push publicly. ::
113
114 $ echo 'debug hack' >> file1.c
115 $ hg commit -m 'debug hack'
116
117 In either case, ``hg prune`` is the answer. ``prune`` simply marks
118 changesets obsolete without creating any new changesets to replace
119 them::
120
121 $ hg prune .
122 1 changesets pruned
123 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
124 working directory now at 934359450037
125
126 Outwardly, it appears that your “debug hack” commit never happened;
127 we're right back where we started::
128
129 $ hg parents --template '{rev}:{node|short} {desc|firstline}\n'
130 3:934359450037 implement feature Y
131
132 In reality, though, the “debug hack” is still there, obsolete and hidden.
133
134 Example 5: Uncommit changes to certain files
135 ============================================
136
137 Occasionally you commit more than you intended: perhaps you made
138 unrelated changes to different files, and thus intend to commit
139 different files separately. ::
140
141 $ echo 'relevant' >> file1.c
142 $ echo 'irrelevant' >> file2.c
143
144 If you forget to specify filenames on the ``commit`` command line,
145 Mercurial commits all those changes together::
146
147 $ hg commit -m 'fix bug 234' # oops: too many files
148
149 Luckily, this mistake is easy to fix with ``uncommit``::
150
151 $ hg uncommit file2.c
152 $ hg status
153 M file2.c
154
155 Let's verify that the replacement changeset looks right (i.e.,
156 modifies only ``file1.c``)::
157
158 $ hg parents --template '{rev}:{node|short} {desc|firstline}\n{files}\n'
159 6:c8defeecf7a4 fix bug 234
160 file1.c
161
162 As before, the original flawed changeset is still there, but obsolete
163 and hidden. It won't be exchanged with other repositories by ``push``,
164 ``pull``, or ``clone``.
165
166 Example 6: Fold multiple changesets together into one
167 =====================================================
168
169 If you're making extensive changes to fragile source code, you might
170 commit more frequently than normal so that you can fallback on a
171 known good state if one step goes badly. ::
172
173 $ echo step1 >> file1.c
174 $ hg commit -m 'step 1' # revision 7
175 $ echo step2 >> file1.c
176 $ hg commit -m 'step 2' # revision 8
177 $ echo step3 >> file2.c
178 $ hg commit -m 'step 3' # revision 9
179
180 At the end of such a sequence, you often end up with a series of small
181 changesets that are tedious to review individually. It might make more
182 sense to combine them into a single changeset using the ``fold``
183 command.
184
185 To make sure we pass the right revisions to ``fold``, let's review the
186 changesets we just created, from revision 7::
187
188 $ hg log --template '{rev}:{node|short} {desc|firstline}\n' -r 7::
189 7:05e61aab8294 step 1
190 8:be6d5bc8e4cc step 2
191 9:35f432d9f7c1 step 3
192
193 and fold them::
194
195 $ hg fold -m 'fix bug 64' -r 7::
196 3 changesets folded
197 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
198
199 This time, Mercurial marks three changesets obsolete, replacing them
200 all with a single *successor*.
201
202 (You might be familiar with this operation under other names, like
203 *squash* or *collapse*.)
204
205 Changeset obsolescence under the hood
206 -------------------------------------
207
208 So far, everything has gone just fine. We haven't run into merge
209 conflicts or other trouble. Before we start exploring advanced usage
210 that can run into trouble, let's step back and see what happens when
211 Mercurial marks changesets obsolete. That will make it much easier to
212 understand the more advanced use cases we'll see later.
213
214 When you have the ``evolve`` extension enabled, all history
215 modification uses the same underlying mechanism: the original
216 changesets are marked *obsolete* and replaced by zero or more
217 *successors*. The obsolete changesets are the *precursors* of their
218 successors. This applies equally to built-in commands (``commit
219 --amend``), commands added by ``evolve`` (``amend``, ``prune``,
220 ``uncommit``, ``fold``), and even commands provided by other
221 extensions (``rebase``, ``histedit``).
222
223 Another way of looking at it is that obsolescence is second-order
224 version control, i.e. the history of your history. We'll cover this in
225 more detail (and mathematical precision) in the `concepts`_ guide.
226
227 .. _`concepts`: concepts.html
228
229 Under the hood: Amend a changeset
230 =================================
231
232 Consider Example 2, amending a changeset with ``evolve``. We saw above
233 that you can do this using the exact same command-line syntax as core
234 Mercurial, namely ``hg commit --amend``. But the implementation is
235 quite different, and Figure 2 shows how.
236
237 .. figure:: figures/figure-ug02.svg
238
239 Figure 2: safe history modification using ``evolve``: the original
240 revision 1 is preserved as an obsolete changeset. (The "temporary
241 amend commit", marked with T, is an implementation detail stemming
242 from limitations in Mercurial's current merge machinery. Future
243 versions of Mercurial will not create them.)
244
245 In this case, the obsolete changesets are also *hidden*. That is the
246 usual end state for obsolete changesets. But many scenarios result in
247 obsolete changesets that are still visible, which indicates your
248 history modification work is not yet done. We'll see examples of that
249 later, when we cover advanced usage.
250
251 Seeing hidden changesets
252 ========================
253
254 TODO
255
256 Under the hood: Prune an unwanted changeset
257 ===========================================
258
259 ``prune`` (example 4 above) is the simplest history modification
260 command provided by ``evolve``. All it does is mark the specified
261 changeset(s) obsolete, with no successor/precursor relationships
262 involved. (If the working directory parent was one of the obsolete
263 changesets, ``prune`` updates back to a suitable ancestor.)
264
265 .. figure:: figures/figure-ug03.svg
266
267 Figure 3: pruning a changeset marks it obsolete with no successors.
268
269 Under the hood: Uncommit changes to certain files
270 =================================================
271
272 In one sense, ``uncommit`` is a simplified version of ``amend``. Like
273 ``amend``, it obsoletes one changeset and leaves it with a single
274 successor. Unlike ``amend``, there is no ugly "temporary amend commit"
275 cluttering up the repository.
276
277 In another sense, ``uncommit`` is the inverse of ``amend``: ``amend``
278 takes any uncommitted changes in the working dir and “adds”
279 them to the working directory's parent changeset. (In reality, of
280 course, it creates a successor changeset, marking the original
281 obsolete.) In contrast, ``uncommit`` takes some changes in the working
282 directory's parent and moves them to the working dir, creating a new
283 successor changeset in the process. Figure 4 illustrates.
284
285 .. figure:: figures/figure-ug04.svg
286
287 Figure 4: uncommit moves some of the changes from the working
288 directory parent into the working dir, preserving the remaining
289 changes as a new successor changeset. (N.B. revision 4 is not shown
290 here because it was marked obsolete in the previous example.)
291
292
293 Under the hood: Fold multiple changesets together into one
294 ==========================================================
295
296 The last basic example is folding multiple changesets into one, which
297 marks multiple changesets obsolete, replacing them all with a single
298 successor.
299
300 .. figure:: figures/figure-ug05.svg
301
302 Figure 5: fold combines multiple changesets into a single
303 successor, marking the original (folded) changesets obsolete.
304
305
306 Obsolete is not hidden
307 ======================
308
309 TODO
310
311
312 Understanding revision numbers
313 ==============================
314
315 If you're trying these examples on your own, especially using ``hg
316 log`` without ``--hidden``, you have probably noticed some funny
317 business going on with revision numbers: there are now gaps in the
318 sequence. That's something you don't see with plain vanilla Mercurial;
319 normally, revision N is always followed by revision N+1.
320
321 This is just the visible manifestation of hidden changesets. If
322 revision 95 is followed by revision 98, that means there are two
323 hidden changesets, 96 and 97, in between.
324
325 Note that changeset IDs are still the permanent, immutable identifier
326 for changesets. Revision numbers are, as ever, a handy shorthand that
327 work in your local repository, but cannot be used across repositories.
328 They also have the useful property of showing when there are hidden
329 changesets lurking under the covers, which is why this document uses
330 revision numbers.
331
332
333 Life with ``evolve`` (advanced usage)
334 -------------------------------------
335
336 Now that you've got a solid understanding of how ``evolve`` works in
337 concert with changeset obsolescence, let's explore some more advanced
338 scenarios. All of these scenarios will involve *unstable* changesets,
339 which is an unavoidable consequence of obsolescence. What really sets
340 ``evolve`` apart from other history modification mechanisms is the
341 fact that it recognizes troubles like unstable changesets and provides
342 a consistent way for you to get out of trouble.
343
344 (Incidentally, there are two other types of trouble that changesets
345 can get into with ``evolve``: they may be *divergent* or *bumped*.
346 Both of those states are more likely to occur when `sharing mutable
347 history`_, so we won't see them in this user guide.)
348
349 .. _`sharing mutable history`: sharing.html
350
351
352 Example 7: Amend an older changeset
353 ===================================
354
355 Sometimes you don't notice your mistakes until after you have
356 committed some new changesets on top of them. ::
357
358 $ hg commit -m 'fix bug 17' # rev 11 (mistake here)
359 $ hg commit -m 'cleanup' # rev 12
360 $ hg commit -m 'feature 23' # rev 13
361
362 Traditionally, your only option is to commit an "oops" changeset that
363 fixes your mistake. That works, of course, but it makes you look bad:
364 you made a mistake, and the record of that mistake is recorded in
365 history for all eternity. (If the mistake was in the commit message,
366 too bad.)
367
368 More subtly, there now exist changesets that are *worse* than what
369 came before—the code no longer builds, the tests don't pass, or
370 similar. Anyone reviewing these patches will waste time noticing the
371 error in the earlier patch, and then the correction later on.
372
373 You can avoid all this by amending the bad changeset and *evolving*
374 subsequent history. Here's how it works, assuming you have just
375 committed revision 13 and noticed the mistake in revision 11::
376
377 $ hg update 11
378 [...fix mistake...]
379 $ hg amend
380
381 At this point, revision 11 is *obsolete* and revisions 12 and 13—the
382 descendants of 11—are in a funny state: they are *unstable*.
383
384 .. figure:: figures/figure-ug06.svg
385
386 Figure 6: amending a changeset with descendants means the amended
387 changeset is obsolete but remains visible; its non-obsolete
388 descendants are *unstable*. The temporary amend commit, revision
389 14, is hidden because it has no non-obsolete descendants.
390
391 All non-obsolete descendants of an obsolete changeset are unstable. An
392 interesting consequence of this is that revision 11 is still visible,
393 even though it is obsolete. Obsolete changesets with non-obsolete
394 descendants are not hidden.
395
396 The fix is to *evolve* history::
397
398 $ hg evolve --all
399
400 This is a separate step, not automatically part of ``hg amend``,
401 because there might be conflicts. If your amended changeset modifies a
402 file that one of its descendants also modified, Mercurial has to fire
403 up your merge tool to resolve the conflict. More importantly, you have
404 to switch contexts from "writing code" to "resolving conflicts". That
405 can be an expensive context switch, so Mercurial lets you decide when
406 to do it.
407
408 The end state, after ``evolve`` finishes, is that the original
409 revisions (11-13) are obsolete and hidden. Their successor revisions
410 (15-17) replace them.
411
412 .. figure:: figures/figure-ug07.svg
413
414 Figure 7: evolve your repository (``hg evolve --all``) to take care
415 of instability. Unstable changesets become obsolete, and are
416 replaced by successors just like the amended changeset was.
417
418 Example 8: Prune an older changeset
419 ===================================
420
421 Let's say you've just committed the following changesets::
422
423 $ hg commit -m 'useful work' # rev 18
424 $ hg commit -m 'debug hack' # rev 19
425 $ hg commit -m 'more work' # rev 20
426
427 You want to drop revision 19, but keep 18 and 20. No problem::
428
429 $ hg prune 19
430 1 changesets pruned
431 1 new unstable changesets
432
433 As above, this leaves your repository in a funny intermediate state:
434 revision 20 is the non-obsolete descendant of obsolete revision 19.
435 That is, revision 20 is unstable.
436
437 .. figure:: figures/figure-ug08.svg
438
439 Figure 8: ``hg prune`` marks a changeset obsolete without creating
440 a successor. Just like with ``hg amend``, non-obsolete descendants
441 of the pruned changeset are now unstable.
442
443 As before, the solution to unstable changesets is to evolve your
444 repository::
445
446 $ hg evolve --all
447
448 This rebases revision 20 on top of 18 as the new revision 21, leaving
449 19 and 20 obsolete and hidden:
450
451 .. figure:: figures/figure-ug09.svg
452
453 Figure 9: once again, ``hg evolve --all`` takes care of instability.
454
455 Example 9: Uncommit files from an older changeset (discard changes)
456 =======================================================================
457
458 As in example 5, let's say you accidentally commit some unrelated
459 changes together. Unlike example 5, you don't notice your mistake
460 immediately, but commit a new changeset on top of the bad one. ::
461
462 $ echo 'this fixes bug 53' >> file1.c
463 $ echo 'debug hack' >> file2.c
464 $ hg commit -m 'fix bug 53' # rev 22 (oops)
465 $ echo 'and this handles bug 67' >> file1.c
466 $ hg commit -m 'fix bug 67' # rev 23 (fine)
467
468 As with ``amend``, you need to travel back in time and repair revision
469 22, leaving your changes to ``file2.c`` back in the working
470 directory::
471
472 $ hg update 22
473 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
474 $ hg uncommit file2.c
475 1 new unstable changesets
476 $ hg status
477 M file2.c
478
479 Now your repository has unstable changesets, so you need to evolve it.
480 But ``hg evolve`` requires a clean working directory to resolve merge
481 conflicts, so you need to decide what to do with ``file2.c``.
482
483 In this case, the change to ``file2.c`` was a temporary debugging
484 hack, so we can discard it and immediately evolve the instability away::
485
486 $ hg revert file2.c
487 $ hg evolve --all
488 move:[23] fix bug 67
489 atop:[24] fix bug 53
490
491 Figure 10 illustrates the whole process.
492
493 .. figure:: figures/figure-ug10.svg
494
495 Figure 10: ``hg uncommit`` of a changeset with descendants results
496 in instability *and* a dirty working directory, both of which must
497 be dealt with.
498
499
500 Example 10: Uncommit files to an older changeset (keep changes)
501 ===================================================================
502
503 This is very similar to example 9. The difference that this time, our
504 change to ``file2.c`` is valuable enough to commit, making things a
505 bit more complicated. The setup is nearly identical::
506
507 $ echo 'fix a bug' >> file1.c
508 $ echo 'useful but unrelated' >> file2.c
509 $ hg commit -u dan -d '11 0' -m 'fix a bug' # rev 26 (oops)
510 $ echo 'new feature' >> file1.c
511 $ hg commit -u dan -d '12 0' -m 'new feature' # rev 27 (fine)
512
513 As before, we update back to the flawed changeset (this time,
514 revision 26) and ``uncommit``, leaving uncommitted changes to
515 ``file2.c`` in the working dir::
516
517 $ hg update -q 26
518 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
519 $ hg uncommit -q file2.c # obsoletes rev 26, creates rev 28
520 1 new unstable changesets
521 $ hg status
522 M file2.c
523
524 This time, let's save that useful change before evolving::
525
526 $ hg commit -m 'useful tweak' # rev 29
527
528 Figure 11 shows the story so far: ``uncommit`` obsoleted revision 26
529 and created revision 28, the successor of 26. Then we committed
530 revision 29, a child of 28. We still have to deal with the unstable
531 revision 27.
532
533 .. figure:: figures/figure-ug11.svg
534
535 Figure 11: Uncommitting a file and then committing that change
536 separately will soon result in a two-headed repository.
537
538 This is where things get tricky. As usual when a repository has
539 unstable changesets, we want to evolve it::
540
541 $ hg evolve --all
542
543 The problem is that ``hg evolve`` rebases revision 27 onto revision
544 28, creating 30 (the successor of 27). This is entirely logical: 27
545 was the child of 26, and 26's successor is 28. So of course 27's
546 successor (30) should be the child of 26's successor (28).
547 Unfortunately, that leaves us with a two-headed repository:
548
549 .. figure:: figures/figure-ug12.svg
550
551 Figure 12: ``evolve`` takes care of unstable changesets; it does
552 not solve all the world's problems.
553
554 As usual when faced with a two-headed repository, you can either merge
555 or rebase. It's up to you.
556
557
558 Example 11: Recover an obsolete changeset
559 =========================================
560
561 TODO