test_hooks.py 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192
  1. # test_hooks.py -- Tests for executing hooks
  2. #
  3. # Dulwich is dual-licensed under the Apache License, Version 2.0 and the GNU
  4. # General Public License as public by the Free Software Foundation; version 2.0
  5. # or (at your option) any later version. You can redistribute it and/or
  6. # modify it under the terms of either of these two licenses.
  7. #
  8. # Unless required by applicable law or agreed to in writing, software
  9. # distributed under the License is distributed on an "AS IS" BASIS,
  10. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  11. # See the License for the specific language governing permissions and
  12. # limitations under the License.
  13. #
  14. # You should have received a copy of the licenses; if not, see
  15. # <http://www.gnu.org/licenses/> for a copy of the GNU General Public License
  16. # and <http://www.apache.org/licenses/LICENSE-2.0> for a copy of the Apache
  17. # License, Version 2.0.
  18. #
  19. """Tests for executing hooks."""
  20. import os
  21. import stat
  22. import shutil
  23. import sys
  24. import tempfile
  25. from dulwich import errors
  26. from dulwich.hooks import (
  27. PreCommitShellHook,
  28. PostCommitShellHook,
  29. CommitMsgShellHook,
  30. )
  31. from dulwich.tests import TestCase
  32. class ShellHookTests(TestCase):
  33. def setUp(self):
  34. super(ShellHookTests, self).setUp()
  35. if os.name != "posix":
  36. self.skipTest("shell hook tests requires POSIX shell")
  37. self.assertTrue(os.path.exists("/bin/sh"))
  38. def test_hook_pre_commit(self):
  39. repo_dir = os.path.join(tempfile.mkdtemp())
  40. os.mkdir(os.path.join(repo_dir, "hooks"))
  41. self.addCleanup(shutil.rmtree, repo_dir)
  42. pre_commit_fail = """#!/bin/sh
  43. exit 1
  44. """
  45. pre_commit_success = """#!/bin/sh
  46. exit 0
  47. """
  48. pre_commit_cwd = (
  49. """#!/bin/sh
  50. if [ "$(pwd)" != '"""
  51. + repo_dir
  52. + """' ]; then
  53. echo "Expected path '"""
  54. + repo_dir
  55. + """', got '$(pwd)'"
  56. exit 1
  57. fi
  58. exit 0
  59. """
  60. )
  61. pre_commit = os.path.join(repo_dir, "hooks", "pre-commit")
  62. hook = PreCommitShellHook(repo_dir, repo_dir)
  63. with open(pre_commit, "w") as f:
  64. f.write(pre_commit_fail)
  65. os.chmod(pre_commit, stat.S_IREAD | stat.S_IWRITE | stat.S_IEXEC)
  66. self.assertRaises(errors.HookError, hook.execute)
  67. if sys.platform != "darwin":
  68. # Don't bother running this test on darwin since path
  69. # canonicalization messages with our simple string comparison.
  70. with open(pre_commit, "w") as f:
  71. f.write(pre_commit_cwd)
  72. os.chmod(pre_commit, stat.S_IREAD | stat.S_IWRITE | stat.S_IEXEC)
  73. hook.execute()
  74. with open(pre_commit, "w") as f:
  75. f.write(pre_commit_success)
  76. os.chmod(pre_commit, stat.S_IREAD | stat.S_IWRITE | stat.S_IEXEC)
  77. hook.execute()
  78. def test_hook_commit_msg(self):
  79. repo_dir = os.path.join(tempfile.mkdtemp())
  80. os.mkdir(os.path.join(repo_dir, "hooks"))
  81. self.addCleanup(shutil.rmtree, repo_dir)
  82. commit_msg_fail = """#!/bin/sh
  83. exit 1
  84. """
  85. commit_msg_success = """#!/bin/sh
  86. exit 0
  87. """
  88. commit_msg_cwd = (
  89. """#!/bin/sh
  90. if [ "$(pwd)" = '"""
  91. + repo_dir
  92. + "' ]; then exit 0; else exit 1; fi\n"
  93. )
  94. commit_msg = os.path.join(repo_dir, "hooks", "commit-msg")
  95. hook = CommitMsgShellHook(repo_dir)
  96. with open(commit_msg, "w") as f:
  97. f.write(commit_msg_fail)
  98. os.chmod(commit_msg, stat.S_IREAD | stat.S_IWRITE | stat.S_IEXEC)
  99. self.assertRaises(errors.HookError, hook.execute, b"failed commit")
  100. if sys.platform != "darwin":
  101. # Don't bother running this test on darwin since path
  102. # canonicalization messages with our simple string comparison.
  103. with open(commit_msg, "w") as f:
  104. f.write(commit_msg_cwd)
  105. os.chmod(commit_msg, stat.S_IREAD | stat.S_IWRITE | stat.S_IEXEC)
  106. hook.execute(b"cwd test commit")
  107. with open(commit_msg, "w") as f:
  108. f.write(commit_msg_success)
  109. os.chmod(commit_msg, stat.S_IREAD | stat.S_IWRITE | stat.S_IEXEC)
  110. hook.execute(b"empty commit")
  111. def test_hook_post_commit(self):
  112. (fd, path) = tempfile.mkstemp()
  113. os.close(fd)
  114. repo_dir = os.path.join(tempfile.mkdtemp())
  115. os.mkdir(os.path.join(repo_dir, "hooks"))
  116. self.addCleanup(shutil.rmtree, repo_dir)
  117. post_commit_success = (
  118. """#!/bin/sh
  119. rm """
  120. + path
  121. + "\n"
  122. )
  123. post_commit_fail = """#!/bin/sh
  124. exit 1
  125. """
  126. post_commit_cwd = (
  127. """#!/bin/sh
  128. if [ "$(pwd)" = '"""
  129. + repo_dir
  130. + "' ]; then exit 0; else exit 1; fi\n"
  131. )
  132. post_commit = os.path.join(repo_dir, "hooks", "post-commit")
  133. hook = PostCommitShellHook(repo_dir)
  134. with open(post_commit, "w") as f:
  135. f.write(post_commit_fail)
  136. os.chmod(post_commit, stat.S_IREAD | stat.S_IWRITE | stat.S_IEXEC)
  137. self.assertRaises(errors.HookError, hook.execute)
  138. if sys.platform != "darwin":
  139. # Don't bother running this test on darwin since path
  140. # canonicalization messages with our simple string comparison.
  141. with open(post_commit, "w") as f:
  142. f.write(post_commit_cwd)
  143. os.chmod(post_commit, stat.S_IREAD | stat.S_IWRITE | stat.S_IEXEC)
  144. hook.execute()
  145. with open(post_commit, "w") as f:
  146. f.write(post_commit_success)
  147. os.chmod(post_commit, stat.S_IREAD | stat.S_IWRITE | stat.S_IEXEC)
  148. hook.execute()
  149. self.assertFalse(os.path.exists(path))