PageRenderTime 57ms CodeModel.GetById 28ms RepoModel.GetById 0ms app.codeStats 0ms

/master/buildbot/test/unit/test_db_migrate_versions_045_worker_transition.py

https://gitlab.com/murder187ss/buildbot
Python | 384 lines | 352 code | 13 blank | 19 comment | 0 complexity | ef3f2ee718723c942a480c52d21766fb 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. import sqlalchemy as sa
  16. from sqlalchemy.engine.reflection import Inspector
  17. from twisted.internet import defer
  18. from twisted.trial import unittest
  19. from buildbot.db.types.json import JsonObject
  20. from buildbot.test.util import migration
  21. from buildbot.util import sautils
  22. class Migration(migration.MigrateTestMixin, unittest.TestCase):
  23. def setUp(self):
  24. return self.setUpMigrateTest()
  25. def tearDown(self):
  26. return self.tearDownMigrateTest()
  27. def _define_old_tables(self, metadata):
  28. self.buildrequests = sautils.Table(
  29. 'buildrequests', metadata,
  30. sa.Column('id', sa.Integer, primary_key=True),
  31. sa.Column('buildsetid', sa.Integer, sa.ForeignKey('buildsets.id'),
  32. nullable=False),
  33. # ...
  34. )
  35. self.buildsets = sautils.Table(
  36. 'buildsets', metadata,
  37. sa.Column('id', sa.Integer, primary_key=True),
  38. sa.Column('parent_buildid', sa.Integer,
  39. sa.ForeignKey('builds.id', use_alter=True, name='parent_buildid')),
  40. # ...
  41. )
  42. self.builders = sautils.Table(
  43. 'builders', metadata,
  44. sa.Column('id', sa.Integer, primary_key=True),
  45. # ...
  46. )
  47. self.builder_masters = sautils.Table(
  48. 'builder_masters', metadata,
  49. sa.Column('id', sa.Integer, primary_key=True, nullable=False),
  50. # ...
  51. )
  52. self.masters = sautils.Table(
  53. "masters", metadata,
  54. sa.Column('id', sa.Integer, primary_key=True),
  55. # ...
  56. )
  57. self.builds = sautils.Table(
  58. 'builds', metadata,
  59. sa.Column('id', sa.Integer, primary_key=True),
  60. sa.Column('number', sa.Integer, nullable=False),
  61. sa.Column('builderid', sa.Integer, sa.ForeignKey('builders.id')),
  62. sa.Column('buildrequestid', sa.Integer, sa.ForeignKey('buildrequests.id'),
  63. nullable=False),
  64. sa.Column('buildslaveid', sa.Integer),
  65. sa.Column('masterid', sa.Integer, sa.ForeignKey('masters.id'),
  66. nullable=False),
  67. sa.Column('started_at', sa.Integer, nullable=False),
  68. sa.Column('complete_at', sa.Integer),
  69. sa.Column('state_string', sa.Text, nullable=False, server_default=''),
  70. sa.Column('results', sa.Integer),
  71. )
  72. self.buildslaves = sautils.Table(
  73. "buildslaves", metadata,
  74. sa.Column("id", sa.Integer, primary_key=True),
  75. sa.Column("name", sa.String(50), nullable=False),
  76. sa.Column("info", JsonObject, nullable=False),
  77. )
  78. self.configured_buildslaves = sautils.Table(
  79. 'configured_buildslaves', metadata,
  80. sa.Column('id', sa.Integer, primary_key=True, nullable=False),
  81. sa.Column('buildermasterid', sa.Integer,
  82. sa.ForeignKey('builder_masters.id'), nullable=False),
  83. sa.Column('buildslaveid', sa.Integer, sa.ForeignKey('buildslaves.id'),
  84. nullable=False),
  85. )
  86. self.connected_buildslaves = sautils.Table(
  87. 'connected_buildslaves', metadata,
  88. sa.Column('id', sa.Integer, primary_key=True, nullable=False),
  89. sa.Column('masterid', sa.Integer,
  90. sa.ForeignKey('masters.id'), nullable=False),
  91. sa.Column('buildslaveid', sa.Integer, sa.ForeignKey('buildslaves.id'),
  92. nullable=False),
  93. )
  94. def _create_tables_thd(self, conn):
  95. metadata = sa.MetaData()
  96. metadata.bind = conn
  97. self._define_old_tables(metadata)
  98. metadata.create_all()
  99. sa.Index('builds_buildrequestid', self.builds.c.buildrequestid).create()
  100. sa.Index('builds_number',
  101. self.builds.c.builderid, self.builds.c.number,
  102. unique=True).create()
  103. sa.Index('builds_buildslaveid', self.builds.c.buildslaveid).create()
  104. sa.Index('builds_masterid', self.builds.c.masterid).create()
  105. sa.Index('buildslaves_name', self.buildslaves.c.name, unique=True).create()
  106. sa.Index('configured_slaves_buildmasterid',
  107. self.configured_buildslaves.c.buildermasterid).create()
  108. sa.Index('configured_slaves_slaves', self.configured_buildslaves.c.buildslaveid).create()
  109. sa.Index('configured_slaves_identity',
  110. self.configured_buildslaves.c.buildermasterid,
  111. self.configured_buildslaves.c.buildslaveid, unique=True).create()
  112. sa.Index('connected_slaves_masterid',
  113. self.connected_buildslaves.c.masterid).create()
  114. sa.Index('connected_slaves_slaves', self.connected_buildslaves.c.buildslaveid).create()
  115. sa.Index('connected_slaves_identity',
  116. self.connected_buildslaves.c.masterid,
  117. self.connected_buildslaves.c.buildslaveid, unique=True).create()
  118. @defer.inlineCallbacks
  119. def test_update_inconsistent_builds_buildslaves(self):
  120. def setup_thd(conn):
  121. self._create_tables_thd(conn)
  122. conn.execute(self.masters.insert(), [
  123. dict(id=1),
  124. dict(id=2),
  125. ])
  126. conn.execute(self.buildsets.insert(), [dict(id=5)])
  127. conn.execute(self.buildrequests.insert(), [
  128. dict(id=3, buildsetid=5),
  129. dict(id=4, buildsetid=5),
  130. ])
  131. conn.execute(self.buildslaves.insert(), [
  132. dict(id=30,
  133. name='worker-1',
  134. info={}),
  135. dict(id=31,
  136. name='worker-2',
  137. info={"a": 1}),
  138. ])
  139. conn.execute(self.builds.insert(), [
  140. dict(id=10,
  141. number=2,
  142. buildrequestid=3,
  143. buildslaveid=123,
  144. masterid=1,
  145. started_at=0,
  146. state_string='state'),
  147. dict(id=11,
  148. number=1,
  149. buildrequestid=4,
  150. buildslaveid=31,
  151. masterid=2,
  152. started_at=1000,
  153. state_string='state2'),
  154. ])
  155. def verify_thd(conn):
  156. metadata = sa.MetaData()
  157. metadata.bind = conn
  158. # Verify database contents.
  159. # 'workers' table contents.
  160. workers = sautils.Table('workers', metadata, autoload=True)
  161. c = workers.c
  162. q = sa.select(
  163. [c.id, c.name, c.info]
  164. ).order_by(c.id)
  165. self.assertEqual(
  166. q.execute().fetchall(), [
  167. (30, u'worker-1', u'{}'),
  168. (31, u'worker-2', u'{"a": 1}'),
  169. ])
  170. # 'builds' table contents.
  171. builds = sautils.Table('builds', metadata, autoload=True)
  172. c = builds.c
  173. q = sa.select(
  174. [c.id, c.number, c.builderid, c.buildrequestid, c.workerid,
  175. c.masterid, c.started_at, c.complete_at, c.state_string,
  176. c.results]
  177. ).order_by(c.id)
  178. # Check that build with invalid reference to buildslaves now
  179. # have no reference to it.
  180. self.assertEqual(
  181. q.execute().fetchall(), [
  182. (10, 2, None, 3, None, 1, 0, None, u'state', None),
  183. (11, 1, None, 4, 31, 2, 1000, None, u'state2', None),
  184. ])
  185. yield self.do_test_migration(44, 45, setup_thd, verify_thd)
  186. def test_update(self):
  187. def setup_thd(conn):
  188. self._create_tables_thd(conn)
  189. conn.execute(self.masters.insert(), [
  190. dict(id=10),
  191. dict(id=11),
  192. ])
  193. conn.execute(self.buildsets.insert(), [
  194. dict(id=90),
  195. dict(id=91),
  196. ])
  197. conn.execute(self.buildrequests.insert(), [
  198. dict(id=20, buildsetid=90),
  199. dict(id=21, buildsetid=91),
  200. ])
  201. conn.execute(self.builders.insert(), [
  202. dict(id=50)
  203. ])
  204. conn.execute(self.buildslaves.insert(), [
  205. dict(id=30,
  206. name='worker-1',
  207. info={}),
  208. dict(id=31,
  209. name='worker-2',
  210. info={"a": 1}),
  211. ])
  212. conn.execute(self.builds.insert(), [
  213. dict(id=40,
  214. number=1,
  215. buildrequestid=20,
  216. buildslaveid=30,
  217. masterid=10,
  218. started_at=1000,
  219. state_string='state'),
  220. ])
  221. conn.execute(self.builds.insert(), [
  222. dict(id=41,
  223. number=2,
  224. builderid=50,
  225. buildrequestid=21,
  226. masterid=11,
  227. started_at=2000,
  228. complete_at=3000,
  229. state_string='state 2',
  230. results=9),
  231. ])
  232. conn.execute(self.builder_masters.insert(), [
  233. dict(id=70),
  234. dict(id=71),
  235. ])
  236. conn.execute(self.configured_buildslaves.insert(), [
  237. dict(id=60,
  238. buildermasterid=70,
  239. buildslaveid=30),
  240. dict(id=61,
  241. buildermasterid=71,
  242. buildslaveid=31),
  243. ])
  244. conn.execute(self.connected_buildslaves.insert(), [
  245. dict(id=80,
  246. masterid=10,
  247. buildslaveid=30),
  248. dict(id=81,
  249. masterid=11,
  250. buildslaveid=31),
  251. ])
  252. def verify_thd(conn):
  253. metadata = sa.MetaData()
  254. metadata.bind = conn
  255. # Verify database contents.
  256. # 'workers' table contents.
  257. workers = sautils.Table('workers', metadata, autoload=True)
  258. c = workers.c
  259. q = sa.select(
  260. [c.id, c.name, c.info]
  261. ).order_by(c.id)
  262. self.assertEqual(
  263. q.execute().fetchall(), [
  264. (30, u'worker-1', u'{}'),
  265. (31, u'worker-2', u'{"a": 1}'),
  266. ])
  267. # 'builds' table contents.
  268. builds = sautils.Table('builds', metadata, autoload=True)
  269. c = builds.c
  270. q = sa.select(
  271. [c.id, c.number, c.builderid, c.buildrequestid, c.workerid,
  272. c.masterid, c.started_at, c.complete_at, c.state_string,
  273. c.results]
  274. ).order_by(c.id)
  275. self.assertEqual(
  276. q.execute().fetchall(), [
  277. (40, 1, None, 20, 30, 10, 1000, None, u'state', None),
  278. (41, 2, 50, 21, None, 11, 2000, 3000, u'state 2', 9),
  279. ])
  280. # 'configured_workers' table contents.
  281. configured_workers = sautils.Table('configured_workers', metadata, autoload=True)
  282. c = configured_workers.c
  283. q = sa.select(
  284. [c.id, c.buildermasterid, c.workerid]
  285. ).order_by(c.id)
  286. self.assertEqual(
  287. q.execute().fetchall(), [
  288. (60, 70, 30),
  289. (61, 71, 31),
  290. ])
  291. # 'connected_workers' table contents.
  292. connected_workers = sautils.Table(
  293. 'connected_workers', metadata, autoload=True)
  294. c = connected_workers.c
  295. q = sa.select(
  296. [c.id, c.masterid, c.workerid]
  297. ).order_by(c.id)
  298. self.assertEqual(
  299. q.execute().fetchall(), [
  300. (80, 10, 30),
  301. (81, 11, 31),
  302. ])
  303. # Verify that there is no "slave"-named items in schema.
  304. inspector = Inspector(conn)
  305. def check_name(name, table_name, item_type):
  306. if not name:
  307. return
  308. self.assertTrue(
  309. u"slave" not in name.lower(),
  310. msg=u"'slave'-named {type} in table '{table}': "
  311. u"'{name}'".format(
  312. type=item_type, table=table_name,
  313. name=name))
  314. # Check every table.
  315. for table_name in inspector.get_table_names():
  316. # Check table name.
  317. check_name(table_name, table_name, u"table name")
  318. # Check column names.
  319. for column_info in inspector.get_columns(table_name):
  320. check_name(column_info['name'], table_name, u"column")
  321. # Check foreign key names.
  322. for fk_info in inspector.get_foreign_keys(table_name):
  323. check_name(fk_info['name'], table_name, u"foreign key")
  324. # Check indexes names.
  325. for index_info in inspector.get_indexes(table_name):
  326. check_name(index_info['name'], table_name, u"index")
  327. # Check primary keys constraints names.
  328. pk_info = inspector.get_pk_constraint(table_name)
  329. check_name(pk_info.get('name'), table_name, u"primary key")
  330. # Test that no "slave"-named items present in schema
  331. for name in inspector.get_schema_names():
  332. self.assertTrue(u"slave" not in name.lower())
  333. return self.do_test_migration(44, 45, setup_thd, verify_thd)