2-change-file.txt 2.9 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788
  1. Changing a File and Commit it
  2. =============================
  3. Now we have a first commit, the next one will show a difference.
  4. As seen in the introduction, it's about making a path in a tree point to a
  5. new blob. The old blob will remain to compute the diff. The tree is altered
  6. and the new commit'task is to point to this new version.
  7. In the following examples, we will need a ``repo`` and an initial commit
  8. (``c1``) similar to the one made in the previous chapter::
  9. >>> from dulwich.repo import Repo
  10. >>> from dulwich.objects import Blob, Tree
  11. >>> repo = Repo.init("myrepo", mkdir=True)
  12. >>> blob = Blob.from_string("My file content\n")
  13. >>> tree = Tree()
  14. >>> tree.add(0100644, "spam", blob.id)
  15. >>> repo.object_store.add_object(blob)
  16. >>> repo.object_store.add_object(tree)
  17. >>> c1_id = repo.do_commit("Initial commit.",
  18. ... committer="JaneExample <jane@example.com>", tree=tree.id)
  19. Let's first build the blob::
  20. >>> from dulwich.objects import Blob
  21. >>> spam = Blob.from_string("My new file content\n")
  22. >>> spam.id
  23. '16ee2682887a962f854ebd25a61db16ef4efe49f'
  24. An alternative is to alter the previously constructed blob object::
  25. >>> blob.data = "My new file content\n"
  26. >>> blob.id
  27. '16ee2682887a962f854ebd25a61db16ef4efe49f'
  28. In any case, update the blob id known as "spam". You also have the
  29. opportunity of changing its mode::
  30. >>> tree["spam"] = (0100644, spam.id)
  31. Now let's record the change::
  32. >>> from dulwich.objects import Commit
  33. >>> from time import time
  34. >>> c2 = Commit()
  35. >>> c2.tree = tree.id
  36. >>> c2.parents = [c1_id]
  37. >>> c2.author = c2.committer = "John Doe <john@example.com>"
  38. >>> c2.commit_time = c2.author_time = int(time())
  39. >>> c2.commit_timezone = c2.author_timezone = 0
  40. >>> c2.encoding = "UTF-8"
  41. >>> c2.message = 'Changing "spam"'
  42. In this new commit we record the changed tree id, and most important, the
  43. previous commit as the parent. Parents are actually a list because a commit
  44. may happen to have several parents after merging branches.
  45. Remain to record this whole new family::
  46. >>> repo.object_store.add_object(spam)
  47. >>> repo.object_store.add_object(tree)
  48. >>> repo.object_store.add_object(c2)
  49. You can already ask git to introspect this commit using ``git show`` and the
  50. value of ``c2.id`` as an argument. You'll see the difference will the
  51. previous blob recorded as "spam".
  52. The diff between the previous head and the new one can be printed using
  53. write_tree_diff::
  54. >>> from dulwich.patch import write_tree_diff
  55. >>> import sys
  56. >>> write_tree_diff(sys.stdout, repo.object_store, repo[c1_id].tree, tree.id)
  57. diff --git a/spam b/spam
  58. index c55063a..16ee268 100644
  59. --- a/spam
  60. +++ b/spam
  61. @@ -1,1 +1,1 @@
  62. -My file content
  63. +My new file content
  64. You won't see it using git log because the head is still the previous
  65. commit. It's easy to remedy::
  66. >>> repo.refs['refs/heads/master'] = c2.id
  67. Now all git tools will work as expected.