1A short git tutorial 2==================== 3May 2005 4 5 6Introduction 7------------ 8 9This is trying to be a short tutorial on setting up and using a git 10archive, mainly because being hands-on and using explicit examples is 11often the best way of explaining what is going on. 12 13In normal life, most people wouldn't use the "core" git programs 14directly, but rather script around them to make them more palatable. 15Understanding the core git stuff may help some people get those scripts 16done, though, and it may also be instructive in helping people 17understand what it is that the higher-level helper scripts are actually 18doing. 19 20The core git is often called "plumbing", with the prettier user 21interfaces on top of it called "porcelain". You may want to know what 22the plumbing does for when the porcelain isn't flushing... 23 24 25Creating a git archive 26---------------------- 27 28Creating a new git archive couldn't be easier: all git archives start 29out empty, and the only thing you need to do is find yourself a 30subdirectory that you want to use as a working tree - either an empty 31one for a totally new project, or an existing working tree that you want 32to import into git. 33 34For our first example, we're going to start a totally new archive from 35scratch, with no pre-existing files, and we'll call it "git-tutorial". 36To start up, create a subdirectory for it, change into that 37subdirectory, and initialize the git infrastructure with "git-init-db": 38 39 mkdir git-tutorial 40 cd git-tutorial 41 git-init-db 42 43to which git will reply 44 45 defaulting to local storage area 46 47which is just git's way of saying that you haven't been doing anything 48strange, and that it will have created a local .git directory setup for 49your new project. You will now have a ".git" directory, and you can 50inspect that with "ls". For your new empty project, ls should show you 51three entries: 52 53 - a symlink called HEAD, pointing to "refs/heads/master" 54 55 Don't worry about the fact that the file that the HEAD link points to 56 doesn't even exist yet - you haven't created the commit that will 57 start your HEAD development branch yet. 58 59 - a subdirectory called "objects", which will contain all the git SHA1 60 objects of your project. You should never have any real reason to 61 look at the objects directly, but you might want to know that these 62 objects are what contains all the real _data_ in your repository. 63 64 - a subdirectory called "refs", which contains references to objects. 65 66 In particular, the "refs" subdirectory will contain two other 67 subdirectories, named "heads" and "tags" respectively. They do 68 exactly what their names imply: they contain references to any number 69 of different "heads" of development (aka "branches"), and to any 70 "tags" that you have created to name specific versions of your 71 repository. 72 73 One note: the special "master" head is the default branch, which is 74 why the .git/HEAD file was created as a symlink to it even if it 75 doesn't yet exist. Basically, the HEAD link is supposed to always 76 point to the branch you are working on right now, and you always 77 start out expecting to work on the "master" branch. 78 79 However, this is only a convention, and you can name your branches 80 anything you want, and don't have to ever even _have_ a "master" 81 branch. A number of the git tools will assume that .git/HEAD is 82 valid, though. 83 84 [ Implementation note: an "object" is identified by its 160-bit SHA1 85 hash, aka "name", and a reference to an object is always the 40-byte 86 hex representation of that SHA1 name. The files in the "refs" 87 subdirectory are expected to contain these hex references (usually 88 with a final '\n' at the end), and you should thus expect to see a 89 number of 41-byte files containing these references in this refs 90 subdirectories when you actually start populating your tree ] 91 92You have now created your first git archive. Of course, since it's 93empty, that's not very useful, so let's start populating it with data. 94 95 96 Populating a git archive 97 ------------------------ 98 99We'll keep this simple and stupid, so we'll start off with populating a 100few trivial files just to get a feel for it. 101 102Start off with just creating any random files that you want to maintain 103in your git archive. We'll start off with a few bad examples, just to 104get a feel for how this works: 105 106 echo "Hello World" > a 107 echo "Silly example" > b 108 109you have now created two files in your working directory, but to 110actually check in your hard work, you will have to go through two steps: 111 112 - fill in the "cache" aka "index" file with the information about your 113 working directory state 114 115 - commit that index file as an object. 116 117The first step is trivial: when you want to tell git about any changes 118to your working directory, you use the "git-update-cache" program. That 119program normally just takes a list of filenames you want to update, but 120to avoid trivial mistakes, it refuses to add new entries to the cache 121(or remove existing ones) unless you explicitly tell it that you're 122adding a new entry with the "--add" flag (or removing an entry with the 123"--remove") flag. 124 125So to populate the index with the two files you just created, you can do 126 127 git-update-cache --add a b 128 129and you have now told git to track those two files. 130 131In fact, as you did that, if you now look into your object directory, 132you'll notice that git will have added two new objects to the object 133store. If you did exactly the steps above, you should now be able to do 134 135 ls .git/objects/??/* 136 137and see two files: 138 139 .git/objects/55/7db03de997c86a4a028e1ebd3a1ceb225be238 140 .git/objects/f2/4c74a2e500f5ee1332c86b94199f52b1d1d962 141 142which correspond with the object with SHA1 names of 557db... and f24c7.. 143respectively. 144 145If you want to, you can use "git-cat-file" to look at those objects, but 146you'll have to use the object name, not the filename of the object: 147 148 git-cat-file -t 557db03de997c86a4a028e1ebd3a1ceb225be238 149 150where the "-t" tells git-cat-file to tell you what the "type" of the 151object is. Git will tell you that you have a "blob" object (ie just a 152regular file), and you can see the contents with 153 154 git-cat-file "blob" 557db03de997c86a4a028e1ebd3a1ceb225be238 155 156which will print out "Hello World". The object 557db... is nothing 157more than the contents of your file "a". 158 159[ Digression: don't confuse that object with the file "a" itself. The 160 object is literally just those specific _contents_ of the file, and 161 however much you later change the contents in file "a", the object we 162 just looked at will never change. Objects are immutable. ] 163 164Anyway, as we mentioned previously, you normally never actually take a 165look at the objects themselves, and typing long 40-character hex SHA1 166names is not something you'd normally want to do. The above digression 167was just to show that "git-update-cache" did something magical, and 168actually saved away the contents of your files into the git content 169store. 170 171Updating the cache did something else too: it created a ".git/index" 172file. This is the index that describes your current working tree, and 173something you should be very aware of. Again, you normally never worry 174about the index file itself, but you should be aware of the fact that 175you have not actually really "checked in" your files into git so far, 176you've only _told_ git about them. 177 178However, since git knows about them, you can how start using some of the 179most basic git commands to manipulate the files or look at their status. 180 181In particular, let's not even check in the two files into git yet, we'll 182start off by adding another line to "a" first: 183 184 echo "It's a new day for git" >> a 185 186and you can now, since you told git about the previous state of "a", ask 187git what has changed in the tree compared to your old index, using the 188"git-diff-files" command: 189 190 git-diff-files 191 192oops. That wasn't very readable. It just spit out its own internal 193version of a "diff", but that internal version really just tells you 194that it has noticed that "a" has been modified, and that the old object 195contents it had have been replaced with something else. 196 197To make it readable, we can tell git-diff-files to output the 198differences as a patch, using the "-p" flag: 199 200 git-diff-files -p 201 202which will spit out 203 204 diff --git a/a b/a 205 --- a/a 206 +++ b/a 207 @@ -1 +1,2 @@ 208 Hello World 209 +It's a new day for git 210 211ie the diff of the change we caused by adding another line to "a". 212 213In other words, git-diff-files always shows us the difference between 214what is recorded in the index, and what is currently in the working 215tree. That's very useful. 216 217 218 Committing git state 219 -------------------- 220 221Now, we want to go to the next stage in git, which is to take the files 222that git knows about in the index, and commit them as a real tree. We do 223that in two phases: creating a "tree" object, and committing that "tree" 224object as a "commit" object together with an explanation of what the 225tree was all about, along with information of how we came to that state. 226 227Creating a tree object is trivial, and is done with "git-write-tree". 228There are no options or other input: git-write-tree will take the 229current index state, and write an object that describes that whole 230index. In other words, we're now tying together all the different 231filenames with their contents (and their permissions), and we're 232creating the equivalent of a git "directory" object: 233 234 git-write-tree 235 236and this will just output the name of the resulting tree, in this case 237(if you have does exactly as I've described) it should be 238 239 3ede4ed7e895432c0a247f09d71a76db53bd0fa4 240 241which is another incomprehensible object name. Again, if you want to, 242you can use "git-cat-file -t 3ede4.." to see that this time the object 243is not a "blob" object, but a "tree" object (you can also use 244git-cat-file to actually output the raw object contents, but you'll see 245mainly a binary mess, so that's less interesting). 246 247However - normally you'd never use "git-write-tree" on its own, because 248normally you always commit a tree into a commit object using the 249"git-commit-tree" command. In fact, it's easier to not actually use 250git-write-tree on its own at all, but to just pass its result in as an 251argument to "git-commit-tree". 252 253"git-commit-tree" normally takes several arguments - it wants to know 254what the _parent_ of a commit was, but since this is the first commit 255ever in this new archive, and it has no parents, we only need to pass in 256the tree ID. However, git-commit-tree also wants to get a commit message 257on its standard input, and it will write out the resulting ID for the 258commit to its standard output. 259 260And this is where we start using the .git/HEAD file. The HEAD file is 261supposed to contain the reference to the top-of-tree, and since that's 262exactly what git-commit-tree spits out, we can do this all with a simple 263shell pipeline: 264 265 echo "Initial commit" | git-commit-tree $(git-write-tree) > .git/HEAD 266 267which will say: 268 269 Committing initial tree 3ede4ed7e895432c0a247f09d71a76db53bd0fa4 270 271just to warn you about the fact that it created a totally new commit 272that is not related to anything else. Normally you do this only _once_ 273for a project ever, and all later commits will be parented on top of an 274earlier commit, and you'll never see this "Committing initial tree" 275message ever again. 276 277 278 Making a change 279 --------------- 280 281Remember how we did the "git-update-cache" on file "a" and then we 282changed "a" afterward, and could compare the new state of "a" with the 283state we saved in the index file? 284 285Further, remember how I said that "git-write-tree" writes the contents 286of the _index_ file to the tree, and thus what we just committed was in 287fact the _original_ contents of the file "a", not the new ones. We did 288that on purpose, to show the difference between the index state, and the 289state in the working directory, and how they don't have to match, even 290when we commit things. 291 292As before, if we do "git-diff-files -p" in our git-tutorial project, 293we'll still see the same difference we saw last time: the index file 294hasn't changed by the act of committing anything. However, now that we 295have committed something, we can also learn to use a new command: 296"git-diff-cache". 297 298Unlike "git-diff-files", which showed the difference between the index 299file and the working directory, "git-diff-cache" shows the differences 300between a committed _tree_ and the index file. In other words, 301git-diff-cache wants a tree to be diffed against, and before we did the 302commit, we couldn't do that, because we didn't have anything to diff 303against. 304 305But now we can do 306 307 git-diff-cache -p HEAD 308 309(where "-p" has the same meaning as it did in git-diff-files), and it 310will show us the same difference, but for a totally different reason. 311Now we're not comparing against the index file, we're comparing against 312the tree we just wrote. It just so happens that those two are obviously 313the same. 314 315"git-diff-cache" also has a specific flag "--cached", which is used to 316tell it to show the differences purely with the index file, and ignore 317the current working directory state entirely. Since we just wrote the 318index file to HEAD, doing "git-diff-cache --cached -p HEAD" should thus 319return an empty set of differences, and that's exactly what it does. 320 321However, our next step is to commit the _change_ we did, and again, to 322understand what's going on, keep in mind the difference between "working 323directory contents", "index file" and "committed tree". We have changes 324in the working directory that we want to commit, and we always have to 325work through the index file, so the first thing we need to do is to 326update the index cache: 327 328 git-update-cache a 329 330(note how we didn't need the "--add" flag this time, since git knew 331about the file already). 332 333Note what happens to the different git-diff-xxx versions here. After 334we've updated "a" in the index, "git-diff-files -p" now shows no 335differences, but "git-diff-cache -p HEAD" still _does_ show that the 336current state is different from the state we committed. In fact, now 337"git-diff-cache" shows the same difference whether we use the "--cached" 338flag or not, since now the index is coherent with the working directory. 339 340Now, since we've updated "a" in the index, we can commit the new 341version. We could do it by writing the tree by hand, and committing the 342tree (this time we'd have to use the "-p HEAD" flag to tell commit that 343the HEAD was the _parent_ of the new commit, and that this wasn't an 344initial commit any more), but the fact is, git has a simple helper 345script for doing all of the non-initial commits that does all of this 346for you, and starts up an editor to let you write your commit message 347yourself, so let's just use that: 348 349 git commit 350 351Write whatever message you want, and all the lines that start with '#' 352will be pruned out, and the rest will be used as the commit message for 353the change. If you decide you don't want to commit anything after all at 354this point (you can continue to edit things and update the cache), you 355can just leave an empty message. Otherwise git-commit-script will commit 356the change for you. 357 358(Btw, current versions of git will consider the change in question to be 359so big that it's considered a whole new file, since the diff is actually 360bigger than the file. So the helpful comments that git-commit-script 361tells you for this example will say that you deleted and re-created the 362file "a". For a less contrived example, these things are usually more 363obvious). 364 365You've now made your first real git commit. And if you're interested in 366looking at what git-commit-script really does, feel free to investigate: 367it's a few very simple shell scripts to generate the helpful (?) commit 368message headers, and a few one-liners that actually do the commit itself. 369 370 371 Checking it out 372 --------------- 373 374While creating changes is useful, it's even more useful if you can tell 375later what changed. The most useful command for this is another of the 376"diff" family, namely "git-diff-tree". 377 378git-diff-tree can be given two arbitrary trees, and it will tell you the 379differences between them. Perhaps even more commonly, though, you can 380give it just a single commit object, and it will figure out the parent 381of that commit itself, and show the difference directly. Thus, to get 382the same diff that we've already seen several times, we can now do 383 384 git-diff-tree -p HEAD 385 386(again, "-p" means to show the difference as a human-readable patch), 387and it will show what the last commit (in HEAD) actually changed. 388 389More interestingly, you can also give git-diff-tree the "-v" flag, which 390tells it to also show the commit message and author and date of the 391commit, and you can tell it to show a whole series of diffs. 392Alternatively, you can tell it to be "silent", and not show the diffs at 393all, but just show the actual commit message. 394 395In fact, together with the "git-rev-list" program (which generates a 396list of revisions), git-diff-tree ends up being a veritable fount of 397changes. A trivial (but very useful) script called "git-whatchanged" is 398included with git which does exactly this, and shows a log of recent 399activity. 400 401To see the whole history of our pitiful little git-tutorial project, you 402can do 403 404 git log 405 406which shows just the log messages, or if we want to see the log together 407whith the associated patches use the more complex (and much more 408powerful) 409 410 git-whatchanged -p --root 411 412and you will see exactly what has changed in the repository over its 413short history. 414 415[ Side note: the "--root" flag is a flag to git-diff-tree to tell it to 416 show the initial aka "root" commit too. Normally you'd probably not 417 want to see the initial import diff, but since the tutorial project 418 was started from scratch and is so small, we use it to make the result 419 a bit more interesting ] 420 421With that, you should now be having some inkling of what git does, and 422can explore on your own. 423 424[ to be continued.. cvs2git, tagging versions, branches, merging.. ]