# Git-to-qisrc cookbook ¶

Some companies put all their code in one big VCS repository (a lot of them use Perforce; Facebook and Google use Mercurial). With Git, this is impractical.

qisrc implements a “meta-” version control system, that allows one to manage a collection of Git clones in a big “worktree”. It offers features similar to the “one-big-repository” approach, and if you squint really hard, you can think of the qisrc worktree as a VCS repo.

That means it must be possible to translate Git commands (that you would use if a project is managed using Git without qisrc) to the equivalent qisrc commands.

For the sake of this comparison, let’s suppose that the equivalent of a Git repository is a qisrc group. This makes sense if we suppose that the point of a qisrc group is to work on a project, that for some practical reason can not be kept in only one Git repository.

Let’s also suppose that the manifest is stored in the worktree. In the case of Aldebaran, this means that the worktree must have been created with “-g tools”. If the manifest is not stored in the worktree, you can also have it alongside your worktree by cloning it with Git.

### Git ¶



git clone "git://foo/bar"



### qisrc ¶



qisrc init git@gitlab.aldebaran.lan:qi/manifest.git -g tools -g foo_bar



## Create a project ¶

### Git ¶

Let’s say that we created a repo accessible from the network at  ssh://git@gitlab.aldebaran.lan/nrubinstein/foobar.git  .



mkdir foo/bar
cd foo/bar
git init
git commit --allow-empty -m 'first commit'
git remote add origin git@gitlab.aldebaran.lan:nrubinstein/foobar.git
git push -u origin master



### qisrc ¶

Here, let’s create a Git repository and then create a qisrc group containing only this Git repository. This supposes that you already have a qisrc worktree.

#### Step 1: create the actual Git repository

Create a Git repository that is accessible from the network, like in the ‘Git’ instructions above. Do note that you need to create it locally in your qisrc worktree and to upload a “master” branch. If you do not do this before adding the Git repository to your worktree, qisrc will break.

#### Step 2: add the Git repository to the manifest



cd ../../manifest/default



Edit manifest.xml.

There is a line that says  <remote name="origin" url="ssh://git@gitlab.aldebaran.lan"/>  . This means that we will not have to repeat this part of the repo URL.

Create a line that says  <repo project="nrubinstein/foobar.git" src="foo/bar" remotes="origin"/>  This tag must be a child of the top-level  <manifest>  tag



git commit -am "Add repository foobar"



#### Step 3: create a group in the manifest



<manifest>
...
<groups>
...
<group name="foobar">
<project name="nrubinstein/foobar.git" />
</group>



Notice that the group definition uses the (shortened) URL to the Git repository as seen in the “project” attribute of the “repo” tag, not its path in the worktree as seen in the “src” attribute of the “repo” tag.



git commit -am "Create group foobar"



#### Step 4: push the group



qisrc push



At Aldebaran, we are using code review for the manifest. That means that the group is not usable yet. If the push above does not go to code review, you can skip steps 5 and 6.

#### Step 5: create a branch



git branch foobar
git push -u origin foobar



#### Step 6: checkout the branch



qisrc sync
qisrc checkout foobar



#### Step 7: add the group



qisrc sync



## Create a branch ¶

### Git ¶



git checkout -b foo_bar



### qisrc ¶

#### When creating the branch

##### Step 1: create the branch in the Git repo that you want to change


cd foo/bar
git checkout -b foo_bar


##### Step 2: create the branch in the manifest


cd ../../manifest/default
git checkout -b foo_bar



Then edit manifest.xml: find the line that says  <repo ... src="manifest/default" />  and edit it to say  branch="foo_bar"  . This is not strictly necessary but will make it easier to edit the branch you are working on.

##### Step 3: edit the manifest to switch your repo to your branch

Edit manifest.xml: find the line that says  <repo ... src="foo/bar" />  and edit it to say  branch="foo_bar"  .



git commit -am "Create branch foo_bar"
git push -u origin foo_bar


##### Step 4: checkout your branch for the whole worktree


cd ../..
qisrc sync
qisrc checkout foo_bar



If everything went well, this last command should do nothing.

#### Afterwards

Then, every time you want to change something in a Git repository that you have not changed yet, you have to reproduce step 1 and 3.

## Switch to a branch ¶

### Git ¶



git checkout foo_bar



### qisrc ¶



qisrc checkout foo_bar



## When some changes are integrated ¶

### Git ¶

When using Git, sometimes you create changes on your foobar branch and at some point they are integrated in the master branch. Typically, when that happens, you would remove the foobar branch with:



git checkout foobar
git rebase master
git checkout master
git branch -d foobar



And then you can remove the remote branch with:



git push origin :foobar



### qisrc ¶

Additionally, when using qisrc, there are two scenarios.

#### When some changes on some Git repo have been merged to master

##### Step 1: Delete the branch in the Git repo

First, do the Git steps above as you would normally do, but do not remove the remote branch yet! That would break “qisrc sync” for everyone following your branch.

##### Step 2: Edit the manifest


cd ../../manifest/default



Edit manifest.xml: find the line that says  <repo ... src="foo/bar" branch="foobar" />  and delete the  branch="foobar"  part.



git commit -am "branch foo/bar of foobar has been merged back to master"
git push


##### Step 3: remove the remote branch

Now that the manifest is updated, removing the remote Git branch is safe.



qisrc sync
cd ../../foo/bar
git push origin :foobar


##### Step 4: Checkout your branch for the whole worktree


cd ..
qisrc sync
qisrc checkout foobar



If everything went well, those two commands should not change anything.

#### When all changes in all Git repos have been merged to master

The qisrc manifest branch is useless now, so you can remove it.



qisrc checkout master
cd manifest/default
git push origin :foobar