Differences between revisions 2 and 18 (spanning 16 versions)
Revision 2 as of 2008-04-06 13:18:09
Size: 1083
Editor: PaulMoore
Comment:
Revision 18 as of 2017-04-29 03:31:11
Size: 6123
Editor: mrzv
Comment:
Deletions are marked like this. Additions are marked like this.
Line 1: Line 1:
== Info Extension == ## page was renamed from Source Info Extension
#pragma section-numbers 2
= hg-subtree Extension =

Brief description of extension.

<<TableOfContents>>

== Status ==
Line 5: Line 13:
''Author: Paul Moore'' ''Author: '' ''Dmitriy Morozov''
Line 7: Line 15:
Download site: Download direct from this page - attachment:info.py Repository: ''https://bitbucket.org/mrzv/hg-subtree''
Line 9: Line 17:
=== Overview === Web page: ''https://bitbucket.org/mrzv/hg-subtree''
Line 11: Line 19:
This extension displays basic information about a (local) repository. The information displayed includes:
 * The repository root.
 * The "base hash" - the changeset ID of revision 0 of the repository. This is the nearest equivalent of a unique identifier for the repository.
 * The number of revisions present in the repository.
 * The number of files in the tip revision.
 * The location (if any) from which this repository was cloned.
 * The default push location for the repository.
== Overview ==
Line 19: Line 21:
An example of usage:
{{{
>hg info
Repository: C:\Data\Mercurial\crew
Base Hash: 9117c6561b0bd7792fa13b50d28239d51b78e51f
Revisions: 6469
Files: 787
Cloned From: http://hg.intevation.org/mercurial/crew
}}}
A typical software project has many dependencies. Because these are often shared with other projects, they are version-controlled as their own standalone repositories. But then how can we include these dependencies into the main codebase? Some languages offer central package repositories (e.g., Python's [[https://pypi.python.org/pypi|PyPI]]), which help with dependency management, but in general (e.g., in C++), the problem is acute.
Line 29: Line 23:
=== Configuration === Mercurial provides one solution in its [[https://www.mercurial-scm.org/wiki/Subrepository|Subrepository feature]]. As the linked page suggests, it is considered a "feature of last resort." In practice, it's tricky to use correctly, let alone effortlessly. The hg-subtree extension offers an alternative solution. The contents of the external project are pulled and merged directly into the main repository (with possibly re-organized directory structure). Developers who pull the repository don't need to know anything about this extension. Its main purpose is to simplify updating the dependency by streamlining the pulling, moving, and merging (and, optionally, collapsing).

Although the name is an homage to [[https://www.atlassian.com/blog/git/alternatives-to-git-submodule-git-subtree|git's subtree feature]], it is a bit of a misnomer, since this extension does not require that subrepository's contents live in a subtree of the main repository. The files can be reorganized to go anywhere.

The hg-subtree extension provides the {{{subpull}}} command.

== Configuration ==
Line 34: Line 34:
info = path/to/info.py subtree = .../hg-subtree/hgsubtree

[subtree]
hgsubtree = .hgsubtree
bookmark = subtree@
move = subtree: move {name}
merge = subtree: update {name}
collapse = subtree: {name}@{rev}
Line 37: Line 44:
== Setup ==

The hg-subtree extension reads configuration of subrepositories from {{{.hgsubtree}}} file (the name is configurable in {{{.hgrc}}}). Each subrepository is its own section. The two main options to set are source and destination:

 * source is any valid repository path; updates are pulled from there using {{{hg pull -f}}}.
 * destination specifies a sequence of commands to transform the contents of the subrepo into a form mergeable into the main repository. In the following example, everything is simply moved into {{{ext/sub}}} subdirectory. Currently, the following commands are supported: {{{mkdir}}}, {{{mv}}}, {{{cp}}}. Any file not touched by the directions in destination gets removed.
{{{
[sub]
source = ssh://.../path/to/sub
destination:
    mkdir ext/sub
    mv * ext/sub
}}}

Optionally:
{{{
collapse = True
rev = master
}}}
{{{collapse}}} is explained below. {{{rev}}} specifies which revision to pull.


== Usage ==

The extension provides only one command: {{{subpull}}}. By default, this command goes through every subrepo specified in {{{.hgsubtree}}}. It pulls the subrepo's contents, applies transformations prescribed by destination, commits those transformations, and then merges the resulting commit with the parent of the working directory.

For example, suppose we have unrelated repositories main and sub.
{{{
$ hg log -R main
@ 1 Add .hgsubtree
|
o 0 Initial commit

$ hg log -R sub
@ 0 Add a.txt

$ ls sub
a.txt
}}}

If main contains {{{.hgsubtree}}} as above, we can {{{subpull}}} from main:
{{{
$ cd main
$ hg subpull
...
$ hg log
@ 4 subtree: update sub
|\
| o 3 subtree: move sub
| |
| o 2 Add a.txt
|
o 1 Add .hgsubtree
|
o 0 Initial commit

$ ls . ext/sub
.:
ext main.txt

ext/sub:
a.txt
}}}

Now if we add another commit to sub (adding file b.txt):
{{{
$ hg log -R sub
@ 1 Add b.txt
|
o 0 Add a.txt
}}}
And subpull from main:
{{{
$ hg subpull
...
$ hg log
@ 7 subtree: update sub
|\
| o 6 subtree: move sub
| \
| o 5 Add b.txt
| |
o | 4 subtree: update sub
|\ |
| o | 3 subtree: move sub
| \|
| o 2 Add a.txt
|
o 1 Add .hgsubtree
|
o 0 Initial commit

$ ls . ext/sub
.:
ext main.txt

ext/sub:
a.txt b.txt
}}}
We get the expected update.

It's possible to pass arguments to subpull to specify pulling from only one of the subrepos in {{{.hgsubtree}}}, or to pull a specific revision ({{{-r}}}), or to override source ({{{-s}}}), or to invoke editor on every commit to modify the messages ({{{-e}}}).

=== Collapse ===
An obvious downside of the above approach is that it could import a lot of history. If the subrepository is your own project, and you plan to push changes back, keeping the full history is a good idea. But if the project is an external dependency, its history would mostly pollute your main repository. To address this problem, it's possible to add {{{collapse = True}}} to any subrepository's section in {{{.hgsubtree}}}. Doing so would collapse all the imported changesets into one. In the previous example, we would get the same contents, but the following history, after the first subpull:
{{{
@ 4 subtree: update sub
|\
| o 3 subtree: move sub
| |
| o 2 subtree: sub@aded622672dd [subtree@sub]
|
o 1 Add .hgsubtree
|
o 0 Initial commit
}}}

And after the second subpull:
{{{
@ 7 subtree: update sub
|\
| o 6 subtree: move sub
| \
| o 5 subtree: sub@85f971e8c898 [subtree@sub]
| |
o | 4 subtree: update sub
|\ |
| o | 3 subtree: move sub
| \|
| o 2 subtree: sub@aded622672dd
|
o 1 Add .hgsubtree
|
o 0 Initial commit
}}}

Notice that {{{subtree@sub}}} bookmark is used to keep track of which revision corresponds to the subrepo. The bookmark prefix is configurable in {{{.hgrc}}}.
Line 38: Line 183:
CategoryExtension CategoryExtensionsByOthers

hg-subtree Extension

Brief description of extension.

1. Status

This extension is not distributed with Mercurial.

Author: Dmitriy Morozov

Repository: https://bitbucket.org/mrzv/hg-subtree

Web page: https://bitbucket.org/mrzv/hg-subtree

2. Overview

A typical software project has many dependencies. Because these are often shared with other projects, they are version-controlled as their own standalone repositories. But then how can we include these dependencies into the main codebase? Some languages offer central package repositories (e.g., Python's PyPI), which help with dependency management, but in general (e.g., in C++), the problem is acute.

Mercurial provides one solution in its Subrepository feature. As the linked page suggests, it is considered a "feature of last resort." In practice, it's tricky to use correctly, let alone effortlessly. The hg-subtree extension offers an alternative solution. The contents of the external project are pulled and merged directly into the main repository (with possibly re-organized directory structure). Developers who pull the repository don't need to know anything about this extension. Its main purpose is to simplify updating the dependency by streamlining the pulling, moving, and merging (and, optionally, collapsing).

Although the name is an homage to git's subtree feature, it is a bit of a misnomer, since this extension does not require that subrepository's contents live in a subtree of the main repository. The files can be reorganized to go anywhere.

The hg-subtree extension provides the subpull command.

3. Configuration

Configure your .hgrc to enable the extension by adding following lines:

[extensions]
subtree = .../hg-subtree/hgsubtree

[subtree]
hgsubtree = .hgsubtree
bookmark = subtree@
move     = subtree: move {name}
merge    = subtree: update {name}
collapse = subtree: {name}@{rev}

4. Setup

The hg-subtree extension reads configuration of subrepositories from .hgsubtree file (the name is configurable in .hgrc). Each subrepository is its own section. The two main options to set are source and destination:

  • source is any valid repository path; updates are pulled from there using hg pull -f.

  • destination specifies a sequence of commands to transform the contents of the subrepo into a form mergeable into the main repository. In the following example, everything is simply moved into ext/sub subdirectory. Currently, the following commands are supported: mkdir, mv, cp. Any file not touched by the directions in destination gets removed.

[sub]
source = ssh://.../path/to/sub
destination:
    mkdir ext/sub
    mv * ext/sub

Optionally:

collapse = True
rev = master

collapse is explained below. rev specifies which revision to pull.

5. Usage

The extension provides only one command: subpull. By default, this command goes through every subrepo specified in .hgsubtree. It pulls the subrepo's contents, applies transformations prescribed by destination, commits those transformations, and then merges the resulting commit with the parent of the working directory.

For example, suppose we have unrelated repositories main and sub.

$ hg log -R main
@  1 Add .hgsubtree
|
o  0 Initial commit

$ hg log -R sub
@  0 Add a.txt

$ ls sub
a.txt

If main contains .hgsubtree as above, we can subpull from main:

$ cd main
$ hg subpull
...
$ hg log
@    4 subtree: update sub
|\
| o  3 subtree: move sub
| |
| o  2 Add a.txt
|
o  1 Add .hgsubtree
|
o  0 Initial commit

$ ls . ext/sub
.:
ext      main.txt

ext/sub:
a.txt

Now if we add another commit to sub (adding file b.txt):

$ hg log -R sub
@  1 Add b.txt
|
o  0 Add a.txt

And subpull from main:

$ hg subpull
...
$ hg log
@    7 subtree: update sub
|\
| o  6 subtree: move sub
|  \
|   o  5 Add b.txt
|   |
o   |  4 subtree: update sub
|\  |
| o |  3 subtree: move sub
|  \|
|   o  2 Add a.txt
|
o  1 Add .hgsubtree
|
o  0 Initial commit

$ ls . ext/sub
.:
ext      main.txt

ext/sub:
a.txt b.txt

We get the expected update.

It's possible to pass arguments to subpull to specify pulling from only one of the subrepos in .hgsubtree, or to pull a specific revision (-r), or to override source (-s), or to invoke editor on every commit to modify the messages (-e).

5.1. Collapse

An obvious downside of the above approach is that it could import a lot of history. If the subrepository is your own project, and you plan to push changes back, keeping the full history is a good idea. But if the project is an external dependency, its history would mostly pollute your main repository. To address this problem, it's possible to add collapse = True to any subrepository's section in .hgsubtree. Doing so would collapse all the imported changesets into one. In the previous example, we would get the same contents, but the following history, after the first subpull:

@    4 subtree: update sub
|\
| o  3 subtree: move sub
| |
| o  2 subtree: sub@aded622672dd [subtree@sub]
|
o  1 Add .hgsubtree
|
o  0 Initial commit

And after the second subpull:

@    7 subtree: update sub
|\
| o  6 subtree: move sub
|  \
|   o  5 subtree: sub@85f971e8c898 [subtree@sub]
|   |
o   |  4 subtree: update sub
|\  |
| o |  3 subtree: move sub
|  \|
|   o  2 subtree: sub@aded622672dd
|
o  1 Add .hgsubtree
|
o  0 Initial commit

Notice that subtree@sub bookmark is used to keep track of which revision corresponds to the subrepo. The bookmark prefix is configurable in .hgrc.


CategoryExtensionsByOthers

Extension Template (last edited 2017-04-29 03:38:34 by mrzv)