1-initial-commit.txt 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119
  1. The Repository
  2. ==============
  3. After this introduction, let's start directly with code::
  4. >>> from dulwich.repo import Repo
  5. The access to every object is through the Repo object. You can open an
  6. existing repository or you can create a new one. There are two types of Git
  7. repositories:
  8. Regular Repositories -- They are the ones you create using ``git init`` and
  9. you daily use. They contain a ``.git`` folder.
  10. Bare Repositories -- There is not ".git" folder. The top-level folder
  11. contains itself the "branches", "hooks"... folders. These are used for
  12. published repositories (mirrors).
  13. Let's create a folder and turn it into a repository, like ``git init`` would::
  14. >>> from os import mkdir
  15. >>> mkdir("myrepo")
  16. >>> repo = Repo.init("myrepo")
  17. >>> repo
  18. <Repo at '/tmp/myrepo/'>
  19. You can already look a the structure of the "myrepo/.git" folder, though it
  20. is mostly empty for now.
  21. Initial commit
  22. ==============
  23. When you use Git, you generally add or modify content. As our repository is
  24. empty for now, we'll start by adding a new file::
  25. >>> from dulwich.objects import Blob
  26. >>> blob = Blob.from_string("My file content\n")
  27. >>> blob.id
  28. 'c55063a4d5d37aa1af2b2dad3a70aa34dae54dc6'
  29. Of course you could create a blob from an existing file using ``from_file``
  30. instead.
  31. As said in the introduction, file content is separed from file name. Let's
  32. give this content a name::
  33. >>> from dulwich.objects import Tree
  34. >>> tree = Tree()
  35. >>> tree.add(0100644, "spam", blob.id)
  36. Note that "0100644" is the octal form for a regular file with common
  37. permissions. You can hardcode them or you can use the ``stat`` module.
  38. The tree state of our repository still needs to be placed in time. That's the
  39. job of the commit::
  40. >>> from dulwich.objects import Commit
  41. >>> from time import time
  42. >>> commit = Commit()
  43. >>> commit.tree = tree.id
  44. >>> author = "Your Name <your.email@example.com>"
  45. >>> commit.author = commit.committer = author
  46. >>> commit.commit_time = commit.author_time = int(time())
  47. >>> tz = parse_timezone('-0200')
  48. >>> commit.commit_timezone = commit.author_timezone = tz
  49. >>> commit.encoding = "UTF-8"
  50. >>> commit.message = "Initial commit"
  51. Note that the initial commit has no parents.
  52. At this point, the repository is still empty because all operations happen in
  53. memory. Let's "commit" it.
  54. >>> object_store = repo.object_store
  55. >>> object_store.add_object(blob)
  56. Now the ".git/objects" folder contains a first SHA-1 file. Let's continue
  57. saving the changes::
  58. >>> object_store.add_object(tree)
  59. >>> object_store.add_object(commit)
  60. Now the physical repository contains three objects but still has no branch.
  61. Let's create the master branch like Git would::
  62. >>> repo.refs['refs/heads/master'] = commit.id
  63. The master branch now has a commit where to start, but Git itself would not
  64. known what is the current branch. That's another reference::
  65. >>> repo.refs['HEAD'] = 'ref: refs/heads/master'
  66. Now our repository is officialy tracking a branch named "master" refering to a
  67. single commit.
  68. Playing again with Git
  69. ======================
  70. At this point you can come back to the shell, go into the "myrepo" folder and
  71. type ``git status`` to let Git confirm that this is a regular repository on
  72. branch "master".
  73. Git will tell you that the file "spam" is deleted, which is normal because
  74. Git is comparing the repository state with the current working copy. And we
  75. have absolutely no working copy using Dulwich because we don't need it at
  76. all!
  77. You can checkout the last state using ``git checkout -f``. The force flag
  78. will prevent Git from complaining that there are uncommitted changes in the
  79. working copy.
  80. The file ``spam`` appears and with no surprise contains the same bytes as the
  81. blob::
  82. $ cat spam
  83. My file content
  84. .. attention:: Remember to recreate the repo object when you modify the
  85. repository outside of Dulwich!