Mercurial > hgbook
view en/branch.tex @ 201:80fc720338a5
Mention use of hooks to defend against propagation of bad changes.
author | Bryan O'Sullivan <bos@serpentine.com> |
---|---|
date | Wed, 18 Apr 2007 15:48:00 -0700 |
parents | 58e3a6c76725 |
children | 6f167e644762 |
line wrap: on
line source
\chapter{Managing releases and branchy development} \label{chap:branch} Mercurial provides several mechanisms for you to manage a project that is making progress on multiple fronts at once. To understand these mechanisms, let's first take a brief look at a fairly normal software project structure. Many software projects issue periodic ``major'' releases that contain substantial new features. In parallel, they may issue ``minor'' releases. These are usually identical to the major releases off which they're based, but with a few bugs fixed. In this chapter, we'll start by talking about how to keep records of project milestones such as releases. We'll then continue on to talk about the flow of work between different phases of a project, and how Mercurial can help you to isolate and manage this work. \section{Giving a persistent name to a revision} Once you decide that you'd like to call a particular revision a ``release'', it's a good idea to record the identity of that revision. This will let you reproduce that release at a later date, for whatever purpose you might need at the time (reproducing a bug, porting to a new platform, etc). \interaction{tag.init} Mercurial lets you give a permanent name to any revision using the \hgcmd{tag} command. Not surprisingly, these names are called ``tags''. \interaction{tag.tag} A tag is nothing more than a ``symbolic name'' for a revision. Tags exist purely for your convenience, so that you have a handy permanent way to refer to a revision; Mercurial doesn't interpret the tag names you use in any way. Neither does Mercurial place any restrictions on the name of a tag, beyond a few that are necessary to ensure that a tag can be parsed unambiguously. A tag name cannot contain any of the following characters: \begin{itemize} \item Colon (ASCII 58, ``\texttt{:}'') \item Carriage return (ASCII 13, ``\texttt{$\backslash$r}'') \item Newline (ASCII 10, ``\texttt{$\backslash$n}'') \end{itemize} You can use the \hgcmd{tags} command to display the tags present in your repository. In the output, each tagged revision is identified first by its name, then by revision number, and finally by the unique hash of the revision. \interaction{tag.tags} Notice that \texttt{tip} is listed in the output of \hgcmd{tags}. The \texttt{tip} tag is a special ``floating'' tag, which always identifies the newest revision in the repository. In the output of the \hgcmd{tags} command, tags are listed in reverse order, by revision number. This usually means that recent tags are listed before older tags. It also means that \texttt{tip} is always going to be the first tag listed in the output of \hgcmd{tags}. When you run \hgcmd{log}, if it displays a revision that has tags associated with it, it will print those tags. \interaction{tag.log} Any time you need to provide a revision~ID to a Mercurial command, the command will accept a tag name in its place. Internally, Mercurial will translate your tag name into the corresponding revision~ID, then use that. \interaction{tag.log.v1.0} There's no limit on the number of tags you can have in a repository, or on the number of tags that a single revision can have. As a practical matter, it's not a great idea to have ``too many'' (a number which will vary from project to project), simply because tags are supposed to help you to find revisions. If you have lots of tags, the ease of using them to identify revisions diminishes rapidly. For example, if your project has milestones as frequent as every few days, it's perfectly reasonable to tag each one of those. But if you have a continuous build system that makes sure every revision can be built cleanly, you'd be introducing a lot of noise if you were to tag every clean build. Instead, you could tag failed builds (on the assumption that they're rare!), or simply not use tags to track buildability. If you want to remove a tag that you no longer want, use \hgcmdargs{tag}{--remove}. \interaction{tag.remove} You can also modify a tag at any time, so that it identifies a different revision, by simply issuing a new \hgcmd{tag} command. You'll have to use the \hgopt{tag}{-f} option to tell Mercurial that you \emph{really} want to update the tag. \interaction{tag.replace} There will still be a permanent record of the previous identity of the tag, but Mercurial will no longer use it. There's thus no penalty to tagging the wrong revision; all you have to do is turn around and tag the correct revision once you discover your error. Mercurial stores tags in a normal revision-controlled file in your repository. If you've created any tags, you'll find them in a file named \sfilename{.hgtags}. When you run the \hgcmd{tag} command, Mercurial modifies this file, then automatically commits the change to it. This means that every time you run \hgcmd{tag}, you'll see a corresponding changeset in the output of \hgcmd{log}. \interaction{tag.tip} \subsection{Handling tag conflicts during a merge} You won't often need to care about the \sfilename{.hgtags} file, but it sometimes makes its presence known during a merge. The format of the file is simple: it consists of a series of lines. Each line starts with a changeset hash, followed by a space, followed by the name of a tag. If you're resolving a conflict in the \sfilename{.hgtags} file during a merge, there's one twist to modifying the \sfilename{.hgtags} file: when Mercurial is parsing the tags in a repository, it \emph{never} reads the working copy of the \sfilename{.hgtags} file. Instead, it reads the \emph{most recently committed} revision of the file. An unfortunate consequence of this design is that you can't actually verify that your merged \sfilename{.hgtags} file is correct until \emph{after} you've committed a change. So if you find yourself resolving a conflict on \sfilename{.hgtags} during a merge, be sure to run \hgcmd{tags} after you commit. If it finds an error in the \sfilename{.hgtags} file, it will report the location of the error, which you can then fix and commit. You should then run \hgcmd{tags} again, just to be sure that your fix is correct. \subsection{Tags and cloning} You may have noticed that the \hgcmd{clone} command has a \hgopt{clone}{-r} option that lets you clone an exact copy of repository as of a particular changeset. The new clone will not contain any project history that comes after the revision you specified. This has an interaction with tags that can surprise the unwary. Recall that a tag is stored as a revision to the \sfilename{.hgtags} file, so that when you create a tag, the changeset in which it's recorded necessarily refers to an older changeset. When you run \hgcmdargs{clone}{-r foo} to clone a repository as of tag \texttt{foo}, the new clone \emph{will not contain the history that created the tag} that you used to clone the repository. The result is that you'll get exactly the right subset of the project's history in the new repository, but \emph{not} the tag you might have expected. \subsection{When permanent tags are too much} Since Mercurial's tags are revision controlled and carried around with a project's history, everyone you work with will see the tags you create. But giving names to revisions has uses beyond simply noting that revision \texttt{4237e45506ee} is really \texttt{v2.0.2}. If you're trying to track down a subtle bug, you might want a tag to remind you of something like ``Anne saw the symptoms with this revision''. For cases like this, what you might want to use are \emph{local} tags. You can create a local tag with the \hgopt{tag}{-l} option to the \hgcmd{tag} command. This will store the tag in a file called \sfilename{.hg/localtags}. Unlike \sfilename{.hgtags}, \sfilename{.hg/localtags} is not revision controlled. Any tags you create using \hgopt{tag}{-l} remain strictly local to the repository you're currently working in. \section{The flow of changes---big picture vs. little} To return to the outline I sketched at the beginning of a chapter, let's think about a project that has multiple concurrent pieces of work under development at once. There might be a push for a new ``main'' release; a new minor bugfix release to the last main release; and an unexpected ``hot fix'' to an old release that is now in maintenance mode. The usual way people refer to these different concurrent directions of development is as ``branches''. However, we've already seen numerous times that Mercurial treats \emph{all of history} as a series of branches and merges. Really, what we have here is two ideas that are peripherally related, but which happen to share a name. \begin{itemize} \item ``Big picture'' branches represent the sweep of a project's evolution; people give them names, and talk about them in conversation. \item ``Little picture'' branches are artefacts of the day-to-day activity of developing and merging changes. They expose the narrative of how the code was developed. \end{itemize} \section{Managing big-picture branches in repositories} The easiest way to isolate a ``big picture'' branch in Mercurial is in a dedicated repository. If you have an existing shared repository---let's call it \texttt{myproject}---that reaches a ``1.0'' milestone, you can start to prepare for future maintenance releases on top of version~1.0 by tagging the revision from which you prepared the~1.0 release. \interaction{branch-repo.tag} You can then clone a new shared \texttt{myproject-1.0.1} repository as of that tag. \interaction{branch-repo.clone} Afterwards, if someone needs to work on a bug fix that ought to go into an upcoming~1.0.1 minor release, they clone the \texttt{myproject-1.0.1} repository, make their changes, and push them back. \interaction{branch-repo.bugfix} Meanwhile, development for the next major release can continue, isolated and unabated, in the \texttt{myproject} repository. \interaction{branch-repo.new} \section{Don't repeat yourself: merging across branches} In many cases, if you have a bug to fix on a maintenance branch, the chances are good that the bug exists on your project's main branch (and possibly other maintenance branches, too). It's a rare developer who wants to fix the same bug multiple times, so let's look at a few ways that Mercurial can help you to manage these bugfixes without duplicating your work. In the simplest instance, all you need to do is pull changes from your maintenance branch into your local clone of the target branch. \interaction{branch-repo.pull} You'll then need to merge the heads of the two branches, and push back to the main branch. \interaction{branch-repo.merge} %%% Local Variables: %%% mode: latex %%% TeX-master: "00book" %%% End: