PageRenderTime 50ms CodeModel.GetById 12ms RepoModel.GetById 0ms app.codeStats 0ms

/worker/buildbot_worker/test/unit/test_bot.py

https://gitlab.com/murder187ss/buildbot
Python | 429 lines | 299 code | 91 blank | 39 comment | 11 complexity | 968de57c162f3f7af7f1eaff293532f9 MD5 | raw file
  1. # This file is part of Buildbot. Buildbot is free software: you can
  2. # redistribute it and/or modify it under the terms of the GNU General Public
  3. # License as published by the Free Software Foundation, version 2.
  4. #
  5. # This program is distributed in the hope that it will be useful, but WITHOUT
  6. # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
  7. # FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
  8. # details.
  9. #
  10. # You should have received a copy of the GNU General Public License along with
  11. # this program; if not, write to the Free Software Foundation, Inc., 51
  12. # Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
  13. #
  14. # Copyright Buildbot Team Members
  15. from builtins import range
  16. import mock
  17. import multiprocessing
  18. import os
  19. import shutil
  20. from twisted.internet import defer
  21. from twisted.internet import reactor
  22. from twisted.internet import task
  23. from twisted.python import failure
  24. from twisted.python import log
  25. from twisted.trial import unittest
  26. import buildbot_worker
  27. from buildbot_worker import base
  28. from buildbot_worker import pb
  29. from buildbot_worker.test.fake.remote import FakeRemote
  30. from buildbot_worker.test.fake.runprocess import Expect
  31. from buildbot_worker.test.util import command
  32. from buildbot_worker.test.util import compat
  33. class TestBot(unittest.TestCase):
  34. def setUp(self):
  35. self.basedir = os.path.abspath("basedir")
  36. if os.path.exists(self.basedir):
  37. shutil.rmtree(self.basedir)
  38. os.makedirs(self.basedir)
  39. self.real_bot = base.BotBase(self.basedir, False)
  40. self.real_bot.startService()
  41. self.bot = FakeRemote(self.real_bot)
  42. def tearDown(self):
  43. d = defer.succeed(None)
  44. if self.real_bot and self.real_bot.running:
  45. d.addCallback(lambda _: self.real_bot.stopService())
  46. if os.path.exists(self.basedir):
  47. shutil.rmtree(self.basedir)
  48. return d
  49. def test_getCommands(self):
  50. d = self.bot.callRemote("getCommands")
  51. def check(cmds):
  52. # just check that 'shell' is present..
  53. self.assertTrue('shell' in cmds)
  54. d.addCallback(check)
  55. return d
  56. def test_getVersion(self):
  57. d = self.bot.callRemote("getVersion")
  58. def check(vers):
  59. self.assertEqual(vers, buildbot_worker.version)
  60. d.addCallback(check)
  61. return d
  62. def test_getSlaveInfo(self):
  63. infodir = os.path.join(self.basedir, "info")
  64. os.makedirs(infodir)
  65. open(os.path.join(infodir, "admin"), "w").write("testy!")
  66. open(os.path.join(infodir, "foo"), "w").write("bar")
  67. open(os.path.join(infodir, "environ"), "w").write("something else")
  68. d = self.bot.callRemote("getSlaveInfo")
  69. def check(info):
  70. self.assertEqual(info, dict(
  71. admin='testy!', foo='bar',
  72. environ=os.environ, system=os.name, basedir=self.basedir,
  73. slave_commands=self.real_bot.remote_getCommands(),
  74. version=self.real_bot.remote_getVersion(),
  75. numcpus=multiprocessing.cpu_count()))
  76. d.addCallback(check)
  77. return d
  78. def test_getSlaveInfo_nodir(self):
  79. d = self.bot.callRemote("getSlaveInfo")
  80. def check(info):
  81. self.assertEqual(set(info.keys()), set(['environ', 'system', 'numcpus', 'basedir', 'slave_commands', 'version']))
  82. d.addCallback(check)
  83. return d
  84. def test_setBuilderList_empty(self):
  85. d = self.bot.callRemote("setBuilderList", [])
  86. def check(builders):
  87. self.assertEqual(builders, {})
  88. d.addCallback(check)
  89. return d
  90. def test_setBuilderList_single(self):
  91. d = self.bot.callRemote("setBuilderList", [('mybld', 'myblddir')])
  92. def check(builders):
  93. self.assertEqual(list(builders), ['mybld'])
  94. self.assertTrue(os.path.exists(os.path.join(self.basedir, 'myblddir')))
  95. # note that we test the WorkerForBuilder instance below
  96. d.addCallback(check)
  97. return d
  98. def test_setBuilderList_updates(self):
  99. d = defer.succeed(None)
  100. workerforbuilders = {}
  101. def add_my(_):
  102. d = self.bot.callRemote("setBuilderList", [
  103. ('mybld', 'myblddir')])
  104. def check(builders):
  105. self.assertEqual(list(builders), ['mybld'])
  106. self.assertTrue(os.path.exists(os.path.join(self.basedir, 'myblddir')))
  107. workerforbuilders['my'] = builders['mybld']
  108. d.addCallback(check)
  109. return d
  110. d.addCallback(add_my)
  111. def add_your(_):
  112. d = self.bot.callRemote("setBuilderList", [
  113. ('mybld', 'myblddir'), ('yourbld', 'yourblddir')])
  114. def check(builders):
  115. self.assertEqual(sorted(builders.keys()), sorted(['mybld', 'yourbld']))
  116. self.assertTrue(os.path.exists(os.path.join(self.basedir, 'myblddir')))
  117. self.assertTrue(os.path.exists(os.path.join(self.basedir, 'yourblddir')))
  118. # 'my' should still be the same WorkerForBuilder object
  119. self.assertEqual(id(workerforbuilders['my']), id(builders['mybld']))
  120. workerforbuilders['your'] = builders['yourbld']
  121. d.addCallback(check)
  122. return d
  123. d.addCallback(add_your)
  124. def remove_my(_):
  125. d = self.bot.callRemote("setBuilderList", [
  126. ('yourbld', 'yourblddir2')]) # note new builddir
  127. def check(builders):
  128. self.assertEqual(sorted(builders.keys()), sorted(['yourbld']))
  129. # note that build dirs are not deleted..
  130. self.assertTrue(os.path.exists(os.path.join(self.basedir, 'myblddir')))
  131. self.assertTrue(os.path.exists(os.path.join(self.basedir, 'yourblddir')))
  132. self.assertTrue(os.path.exists(os.path.join(self.basedir, 'yourblddir2')))
  133. # 'your' should still be the same WorkerForBuilder object
  134. self.assertEqual(id(workerforbuilders['your']), id(builders['yourbld']))
  135. d.addCallback(check)
  136. return d
  137. d.addCallback(remove_my)
  138. def add_and_remove(_):
  139. d = self.bot.callRemote("setBuilderList", [
  140. ('theirbld', 'theirblddir')])
  141. def check(builders):
  142. self.assertEqual(sorted(builders.keys()), sorted(['theirbld']))
  143. self.assertTrue(os.path.exists(os.path.join(self.basedir, 'myblddir')))
  144. self.assertTrue(os.path.exists(os.path.join(self.basedir, 'yourblddir')))
  145. self.assertTrue(os.path.exists(os.path.join(self.basedir, 'theirblddir')))
  146. d.addCallback(check)
  147. return d
  148. d.addCallback(add_and_remove)
  149. return d
  150. def test_shutdown(self):
  151. d1 = defer.Deferred()
  152. self.patch(reactor, "stop", lambda: d1.callback(None))
  153. d2 = self.bot.callRemote("shutdown")
  154. # don't return until both the shutdown method has returned, and
  155. # reactor.stop has been called
  156. return defer.gatherResults([d1, d2])
  157. class FakeStep(object):
  158. "A fake master-side BuildStep that records its activities."
  159. def __init__(self):
  160. self.finished_d = defer.Deferred()
  161. self.actions = []
  162. def wait_for_finish(self):
  163. return self.finished_d
  164. def remote_update(self, updates):
  165. for update in updates:
  166. if 'elapsed' in update[0]:
  167. update[0]['elapsed'] = 1
  168. self.actions.append(["update", updates])
  169. def remote_complete(self, f):
  170. self.actions.append(["complete", f])
  171. self.finished_d.callback(None)
  172. class TestWorkerForBuilder(command.CommandTestMixin, unittest.TestCase):
  173. @defer.inlineCallbacks
  174. def setUp(self):
  175. self.basedir = os.path.abspath("basedir")
  176. if os.path.exists(self.basedir):
  177. shutil.rmtree(self.basedir)
  178. os.makedirs(self.basedir)
  179. self.bot = base.BotBase(self.basedir, False)
  180. self.bot.startService()
  181. # get a WorkerForBuilder object from the bot and wrap it as a fake remote
  182. builders = yield self.bot.remote_setBuilderList([('sb', 'sb')])
  183. self.sb = FakeRemote(builders['sb'])
  184. self.setUpCommand()
  185. def tearDown(self):
  186. self.tearDownCommand()
  187. d = defer.succeed(None)
  188. if self.bot and self.bot.running:
  189. d.addCallback(lambda _: self.bot.stopService())
  190. if os.path.exists(self.basedir):
  191. shutil.rmtree(self.basedir)
  192. return d
  193. def test_print(self):
  194. return self.sb.callRemote("print", "Hello, WorkerForBuilder.")
  195. def test_setMaster(self):
  196. # not much to check here - what the WorkerForBuilder does with the
  197. # master is not part of the interface (and, in fact, it does very little)
  198. return self.sb.callRemote("setMaster", mock.Mock())
  199. def test_shutdown(self):
  200. # don't *actually* shut down the reactor - that would be silly
  201. stop = mock.Mock()
  202. self.patch(reactor, "stop", stop)
  203. d = self.sb.callRemote("shutdown")
  204. def check(_):
  205. self.assertTrue(stop.called)
  206. d.addCallback(check)
  207. return d
  208. def test_startBuild(self):
  209. return self.sb.callRemote("startBuild")
  210. def test_startCommand(self):
  211. # set up a fake step to receive updates
  212. st = FakeStep()
  213. # patch runprocess to handle the 'echo', below
  214. self.patch_runprocess(
  215. Expect(['echo', 'hello'], os.path.join(self.basedir, 'sb', 'workdir'))
  216. + {'hdr': 'headers'} + {'stdout': 'hello\n'} + {'rc': 0}
  217. + 0,
  218. )
  219. d = defer.succeed(None)
  220. def do_start(_):
  221. return self.sb.callRemote("startCommand", FakeRemote(st),
  222. "13", "shell", dict(
  223. command=['echo', 'hello'],
  224. workdir='workdir'))
  225. d.addCallback(do_start)
  226. d.addCallback(lambda _: st.wait_for_finish())
  227. def check(_):
  228. self.assertEqual(st.actions, [
  229. ['update', [[{'hdr': 'headers'}, 0]]],
  230. ['update', [[{'stdout': 'hello\n'}, 0]]],
  231. ['update', [[{'rc': 0}, 0]]],
  232. ['update', [[{'elapsed': 1}, 0]]],
  233. ['complete', None],
  234. ])
  235. d.addCallback(check)
  236. return d
  237. def test_startCommand_interruptCommand(self):
  238. # set up a fake step to receive updates
  239. st = FakeStep()
  240. # patch runprocess to pretend to sleep (it will really just hang forever,
  241. # except that we interrupt it)
  242. self.patch_runprocess(
  243. Expect(['sleep', '10'], os.path.join(self.basedir, 'sb', 'workdir'))
  244. + {'hdr': 'headers'}
  245. + {'wait': True}
  246. )
  247. d = defer.succeed(None)
  248. def do_start(_):
  249. return self.sb.callRemote("startCommand", FakeRemote(st),
  250. "13", "shell", dict(
  251. command=['sleep', '10'],
  252. workdir='workdir'))
  253. d.addCallback(do_start)
  254. # wait a jiffy..
  255. def do_wait(_):
  256. d = defer.Deferred()
  257. reactor.callLater(0.01, d.callback, None)
  258. return d
  259. d.addCallback(do_wait)
  260. # and then interrupt the step
  261. def do_interrupt(_):
  262. return self.sb.callRemote("interruptCommand", "13", "tl/dr")
  263. d.addCallback(do_interrupt)
  264. d.addCallback(lambda _: st.wait_for_finish())
  265. def check(_):
  266. self.assertEqual(st.actions, [
  267. ['update', [[{'hdr': 'headers'}, 0]]],
  268. ['update', [[{'hdr': 'killing'}, 0]]],
  269. ['update', [[{'rc': -1}, 0]]],
  270. ['complete', None],
  271. ])
  272. d.addCallback(check)
  273. return d
  274. def test_startCommand_failure(self):
  275. # set up a fake step to receive updates
  276. st = FakeStep()
  277. # patch runprocess to generate a failure
  278. self.patch_runprocess(
  279. Expect(['sleep', '10'], os.path.join(self.basedir, 'sb', 'workdir'))
  280. + failure.Failure(Exception("Oops"))
  281. )
  282. # patch the log.err, otherwise trial will think something *actually* failed
  283. self.patch(log, "err", lambda f: None)
  284. d = defer.succeed(None)
  285. def do_start(_):
  286. return self.sb.callRemote("startCommand", FakeRemote(st),
  287. "13", "shell", dict(
  288. command=['sleep', '10'],
  289. workdir='workdir'))
  290. d.addCallback(do_start)
  291. d.addCallback(lambda _: st.wait_for_finish())
  292. def check(_):
  293. self.assertEqual(st.actions[1][0], 'complete')
  294. self.assertTrue(isinstance(st.actions[1][1], failure.Failure))
  295. d.addCallback(check)
  296. return d
  297. def test_startCommand_missing_args(self):
  298. # set up a fake step to receive updates
  299. st = FakeStep()
  300. d = defer.succeed(None)
  301. def do_start(_):
  302. return self.sb.callRemote("startCommand", FakeRemote(st),
  303. "13", "shell", dict())
  304. d.addCallback(do_start)
  305. d.addCallback(lambda _: self.assertTrue(False))
  306. d.addErrback(lambda _: True)
  307. return d
  308. class TestBotFactory(unittest.TestCase):
  309. def setUp(self):
  310. self.bf = pb.BotFactory('mstr', 9010, 35, 200)
  311. # tests
  312. def test_timers(self):
  313. clock = self.bf._reactor = task.Clock()
  314. calls = []
  315. def callRemote(method):
  316. calls.append(clock.seconds())
  317. self.assertEqual(method, 'keepalive')
  318. # simulate the response taking a few seconds
  319. d = defer.Deferred()
  320. clock.callLater(5, d.callback, None)
  321. return d
  322. self.bf.perspective = mock.Mock()
  323. self.bf.perspective.callRemote = callRemote
  324. self.bf.startTimers()
  325. clock.callLater(100, self.bf.stopTimers)
  326. clock.pump((1 for _ in range(150)))
  327. self.assertEqual(calls, [35, 70])
  328. @compat.usesFlushLoggedErrors
  329. def test_timers_exception(self):
  330. clock = self.bf._reactor = task.Clock()
  331. self.bf.perspective = mock.Mock()
  332. def callRemote(method):
  333. return defer.fail(RuntimeError("oh noes"))
  334. self.bf.perspective.callRemote = callRemote
  335. self.bf.startTimers()
  336. clock.advance(35)
  337. self.assertEqual(len(self.flushLoggedErrors(RuntimeError)), 1)
  338. # note that the Worker class is tested in test_bot_Worker