mirror of
https://github.com/odoo/runbot.git
synced 2025-04-05 01:31:54 +07:00

On non sticky branches, when a new build is found while another is already testing, the older build is killed. This happens during when the main runbot instance is discovering new commits and create new builds. As a result, concurrent updates may occur while the builders access the concerned build. With this commit, this garbage collecting procedure occurs during the scheduler loop that runs on runbot builder hosts. Also, the logic changed in a way that the kill is requested only if the host needs room to handle pending builds.
988 lines
40 KiB
Python
988 lines
40 KiB
Python
# -*- coding: utf-8 -*-
|
|
import datetime
|
|
|
|
from unittest.mock import patch
|
|
|
|
from odoo import fields
|
|
from .common import RunbotCase
|
|
|
|
|
|
def rev_parse(repo, branch_name):
|
|
"""
|
|
simulate a rev parse by returning a fake hash of form
|
|
'rp_odoo-dev/enterprise_saas-12.2__head'
|
|
should be overwitten if a pr head should match a branch head
|
|
"""
|
|
head_hash = 'rp_%s_%s_head' % (repo.name.split(':')[1], branch_name.split('/')[-1])
|
|
return head_hash
|
|
|
|
|
|
class Test_Build(RunbotCase):
|
|
|
|
def setUp(self):
|
|
super(Test_Build, self).setUp()
|
|
self.repo = self.Repo.create({'name': 'bla@example.com:foo/bar', 'server_files': 'server.py', 'addons_paths': 'addons,core/addons'})
|
|
self.branch = self.Branch.create({
|
|
'repo_id': self.repo.id,
|
|
'name': 'refs/heads/master'
|
|
})
|
|
self.branch_10 = self.Branch.create({
|
|
'repo_id': self.repo.id,
|
|
'name': 'refs/heads/10.0'
|
|
})
|
|
self.branch_11 = self.Branch.create({
|
|
'repo_id': self.repo.id,
|
|
'name': 'refs/heads/11.0'
|
|
})
|
|
self.start_patcher('find_patcher', 'odoo.addons.runbot.common.find', 0)
|
|
|
|
def test_base_fields(self):
|
|
build = self.create_build({
|
|
'branch_id': self.branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'port': '1234',
|
|
})
|
|
self.assertEqual(build.id, build.sequence)
|
|
self.assertEqual(build.dest, '%05d-master-d0d0ca' % build.id)
|
|
# test dest change on new commit
|
|
build.name = 'deadbeef0000ffffffffffffffffffffffffffff'
|
|
self.assertEqual(build.dest, '%05d-master-deadbe' % build.id)
|
|
|
|
# Test domain compute with fqdn and ir.config_parameter
|
|
self.patchers['fqdn_patcher'].return_value = 'runbot98.nowhere.org'
|
|
self.env['ir.config_parameter'].sudo().set_param('runbot.runbot_domain', False)
|
|
self.assertEqual(build.domain, 'runbot98.nowhere.org:1234')
|
|
self.env['ir.config_parameter'].set_param('runbot.runbot_domain', 'runbot99.example.org')
|
|
build._compute_domain()
|
|
self.assertEqual(build.domain, 'runbot99.example.org:1234')
|
|
|
|
# test json stored _data field and data property
|
|
self.assertEqual(build.config_data, {})
|
|
build.config_data = {'restore_url': 'foobar'}
|
|
self.assertEqual(build.config_data, {'restore_url': 'foobar'})
|
|
build.config_data['test_info'] = 'dummy'
|
|
self.assertEqual(build.config_data, {"restore_url": "foobar", "test_info": "dummy"})
|
|
del build.config_data['restore_url']
|
|
self.assertEqual(build.config_data, {"test_info": "dummy"})
|
|
other = self.create_build({
|
|
'branch_id': self.branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'port': '5678',
|
|
'local_result': 'ko'
|
|
})
|
|
|
|
# test a bulk write, that one cannot change from 'ko' to 'ok'
|
|
builds = self.Build.browse([build.id, other.id])
|
|
with self.assertRaises(AssertionError):
|
|
builds.write({'local_result': 'ok'})
|
|
|
|
# test that a build cannot have local state 'duplicate' without a duplicate_id
|
|
with self.assertRaises(AssertionError):
|
|
builds.write({'local_state': 'duplicate'})
|
|
|
|
# test non initialized json stored _data field and data property
|
|
other.config_data['test_info'] = 'foo'
|
|
self.assertEqual(other.config_data, {'test_info': 'foo'})
|
|
(build|other).write({'config_data': {'test_write': 'written'}})
|
|
build.config_data['test_build'] = 'foo'
|
|
other.config_data['test_other'] = 'bar'
|
|
self.assertEqual(build.config_data, {'test_write': 'written', 'test_build': 'foo'})
|
|
self.assertEqual(other.config_data, {'test_write': 'written', 'test_other': 'bar'})
|
|
build.flush()
|
|
build.env.cr.execute("SELECT config_data, config_data->'test_write' AS written, config_data->'test_build' AS test_build FROM runbot_build WHERE id = %s", [build.id])
|
|
self.assertEqual([({'test_write': 'written', 'test_build': 'foo'}, 'written', 'foo')], self.env.cr.fetchall())
|
|
|
|
def test_config_data_duplicate(self):
|
|
|
|
build = self.create_build({
|
|
'branch_id': self.branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
})
|
|
|
|
build2 = self.create_build({
|
|
'branch_id': self.branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
})
|
|
self.assertEqual(build2.duplicate_id, build)
|
|
|
|
build3 = self.create_build({
|
|
'branch_id': self.branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'config_data': {'test':'aa'},
|
|
})
|
|
self.assertFalse(build3.duplicate_id)
|
|
build4 = self.create_build({
|
|
'branch_id': self.branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'config_data': {'test':'aa'},
|
|
})
|
|
self.assertEqual(build4.duplicate_id, build3)
|
|
|
|
build5 = self.create_build({
|
|
'branch_id': self.branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'config_data': {'test':'bb'},
|
|
})
|
|
self.assertFalse(build5.duplicate_id)
|
|
|
|
|
|
@patch('odoo.addons.runbot.models.build.runbot_build._get_repo_available_modules')
|
|
def test_filter_modules(self, mock_get_repo_mods):
|
|
""" test module filtering """
|
|
build = self.create_build({
|
|
'branch_id': self.branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'port': '1234',
|
|
})
|
|
|
|
repo_mods = ['good_module', 'bad_module', 'other_good', 'l10n_be', 'hw_foo', 'hwgood', 'hw_explicit']
|
|
available_mods = ['good_module', 'bad_module', 'other_good', 'l10n_be', 'hw_foo', 'hwgood', 'hw_explicit', 'other_mod_1', 'other_mod_2']
|
|
mock_get_repo_mods.return_value = (repo_mods, available_mods)
|
|
self.repo.modules_auto = 'repo'
|
|
self.repo.modules = '-bad_module,-hw_*,hw_explicit,-l10n_*'
|
|
modules_to_test = build._get_modules_to_test(self, modules_patterns='')
|
|
self.assertEqual(modules_to_test, sorted(['good_module', 'hwgood', 'other_good', 'hw_explicit']))
|
|
|
|
modules_to_test = build._get_modules_to_test(self, modules_patterns='-*, l10n_be')
|
|
self.assertEqual(modules_to_test, sorted(['l10n_be']))
|
|
|
|
modules_to_test = build._get_modules_to_test(self, modules_patterns='l10n_be')
|
|
self.assertEqual(modules_to_test, sorted(['good_module', 'hwgood', 'other_good', 'hw_explicit', 'l10n_be']))
|
|
|
|
# star to get all available mods
|
|
modules_to_test = build._get_modules_to_test(self, modules_patterns='*, -hw_*, hw_explicit')
|
|
self.assertEqual(modules_to_test, sorted(['good_module', 'bad_module', 'other_good', 'l10n_be', 'hwgood', 'hw_explicit', 'other_mod_1', 'other_mod_2']))
|
|
|
|
def test_build_cmd_log_db(self, ):
|
|
""" test that the logdb connection URI is taken from the .odoorc file """
|
|
uri = 'postgres://someone:pass@somewhere.com/db'
|
|
self.env['ir.config_parameter'].sudo().set_param("runbot.runbot_logdb_uri", uri)
|
|
build = self.create_build({
|
|
'branch_id': self.branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'port': '1234',
|
|
})
|
|
cmd = build._cmd(py_version=3)
|
|
self.assertIn('log_db = %s' % uri, cmd.get_config())
|
|
|
|
def test_build_cmd_server_path_no_dep(self):
|
|
""" test that the server path and addons path """
|
|
build = self.create_build({
|
|
'branch_id': self.branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'port': '1234',
|
|
})
|
|
cmd = build._cmd(py_version=3)
|
|
self.assertEqual('python3', cmd[0])
|
|
self.assertEqual('bar/server.py', cmd[1])
|
|
self.assertIn('--addons-path', cmd)
|
|
addons_path_pos = cmd.index('--addons-path') + 1
|
|
self.assertEqual(cmd[addons_path_pos], 'bar/addons,bar/core/addons')
|
|
|
|
def test_build_cmd_server_path_with_dep(self):
|
|
""" test that the server path and addons path """
|
|
|
|
def is_file(file):
|
|
self.assertIn(file, [
|
|
'/tmp/runbot_test/static/sources/bar-ent/d0d0caca0000ffffffffffffffffffffffffffff/requirements.txt',
|
|
'/tmp/runbot_test/static/sources/bar/dfdfcfcf0000ffffffffffffffffffffffffffff/requirements.txt',
|
|
'/tmp/runbot_test/static/sources/bar/dfdfcfcf0000ffffffffffffffffffffffffffff/server.py',
|
|
'/tmp/runbot_test/static/sources/bar/dfdfcfcf0000ffffffffffffffffffffffffffff/openerp/tools/config.py'
|
|
])
|
|
if file == '/tmp/runbot_test/static/sources/bar-ent/d0d0caca0000ffffffffffffffffffffffffffff/requirements.txt':
|
|
return False
|
|
return True
|
|
|
|
def is_dir(file):
|
|
paths = [
|
|
'sources/bar/dfdfcfcf0000ffffffffffffffffffffffffffff/addons',
|
|
'sources/bar/dfdfcfcf0000ffffffffffffffffffffffffffff/core/addons',
|
|
'sources/bar-ent/d0d0caca0000ffffffffffffffffffffffffffff'
|
|
]
|
|
self.assertTrue(any([path in file for path in paths])) # checking that addons path existence check looks ok
|
|
return True
|
|
|
|
self.patchers['isfile'].side_effect = is_file
|
|
self.patchers['isdir'].side_effect = is_dir
|
|
|
|
repo_ent = self.env['runbot.repo'].create({
|
|
'name': 'bla@example.com:foo/bar-ent',
|
|
'server_files': '',
|
|
})
|
|
repo_ent.dependency_ids = self.repo
|
|
enterprise_branch = self.env['runbot.branch'].create({
|
|
'repo_id': repo_ent.id,
|
|
'name': 'refs/heads/master'
|
|
})
|
|
|
|
def rev_parse(repo, branch_name):
|
|
self.assertEqual(repo, self.repo)
|
|
self.assertEqual(branch_name, 'refs/heads/master')
|
|
return 'dfdfcfcf0000ffffffffffffffffffffffffffff'
|
|
|
|
with patch('odoo.addons.runbot.models.repo.runbot_repo._git_rev_parse', new=rev_parse):
|
|
build = self.create_build({
|
|
'branch_id': enterprise_branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'port': '1234',
|
|
})
|
|
cmd = build._cmd(py_version=3)
|
|
self.assertIn('--addons-path', cmd)
|
|
addons_path_pos = cmd.index('--addons-path') + 1
|
|
self.assertEqual(cmd[addons_path_pos], 'bar-ent,bar/addons,bar/core/addons')
|
|
self.assertEqual('bar/server.py', cmd[1])
|
|
self.assertEqual('python3', cmd[0])
|
|
|
|
def test_build_cmd_server_path_with_dep_collision(self):
|
|
""" test that the server path and addons path """
|
|
|
|
def is_file(file):
|
|
self.assertIn(file, [
|
|
'/tmp/runbot_test/static/sources/bar/dfdfcfcf0000ffffffffffffffffffffffffffff/requirements.txt',
|
|
'/tmp/runbot_test/static/sources/bar/d0d0caca0000ffffffffffffffffffffffffffff/requirements.txt',
|
|
'/tmp/runbot_test/static/sources/bar/dfdfcfcf0000ffffffffffffffffffffffffffff/server.py',
|
|
'/tmp/runbot_test/static/sources/bar/dfdfcfcf0000ffffffffffffffffffffffffffff/openerp/tools/config.py'
|
|
])
|
|
if file == '/tmp/runbot_test/static/sources/bar/dfdfcfcf0000ffffffffffffffffffffffffffff/requirements.txt':
|
|
return False
|
|
return True
|
|
|
|
self.patchers['isfile'].side_effect = is_file
|
|
repo_ent = self.env['runbot.repo'].create({
|
|
'name': 'bla@example.com:foo-ent/bar',
|
|
'server_files': '',
|
|
})
|
|
repo_ent.dependency_ids = self.repo
|
|
enterprise_branch = self.env['runbot.branch'].create({
|
|
'repo_id': repo_ent.id,
|
|
'name': 'refs/heads/master'
|
|
})
|
|
|
|
def rev_parse(repo, branch_name):
|
|
self.assertEqual(repo, self.repo)
|
|
self.assertEqual(branch_name, 'refs/heads/master')
|
|
return 'dfdfcfcf0000ffffffffffffffffffffffffffff'
|
|
|
|
with patch('odoo.addons.runbot.models.repo.runbot_repo._git_rev_parse', new=rev_parse):
|
|
build = self.create_build({
|
|
'branch_id': enterprise_branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'port': '1234',
|
|
})
|
|
cmd = build._cmd(py_version=3)
|
|
self.assertIn('--addons-path', cmd)
|
|
addons_path_pos = cmd.index('--addons-path') + 1
|
|
self.assertEqual(cmd[addons_path_pos], 'bar-d0d0caca,bar-dfdfcfcf/addons,bar-dfdfcfcf/core/addons')
|
|
self.assertEqual('bar-dfdfcfcf/server.py', cmd[1])
|
|
self.assertEqual('python3', cmd[0])
|
|
|
|
def test_build_config_from_branch_default(self):
|
|
"""test build config_id is computed from branch default config_id"""
|
|
build = self.create_build({
|
|
'branch_id': self.branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
})
|
|
self.assertEqual(build.config_id, self.env.ref('runbot.runbot_build_config_default'))
|
|
|
|
def test_build_config_from_branch_testing(self):
|
|
"""test build config_id is computed from branch"""
|
|
self.branch.config_id = self.env.ref('runbot.runbot_build_config_default_no_run')
|
|
build = self.create_build({
|
|
'branch_id': self.branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
})
|
|
self.assertEqual(build.config_id, self.branch.config_id, "config_id should be the same as the branch")
|
|
|
|
def test_build_config_can_be_set(self):
|
|
"""test build config_id can be set to something different than the one on the branch"""
|
|
self.branch.config_id = self.env.ref('runbot.runbot_build_config_default')
|
|
build = self.create_build({
|
|
'branch_id': self.branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'config_id': self.env.ref('runbot.runbot_build_config_default_no_run').id
|
|
})
|
|
self.assertEqual(build.config_id, self.env.ref('runbot.runbot_build_config_default_no_run'), "config_id should be the one set on the build")
|
|
|
|
def test_build_gc_date(self):
|
|
""" test build gc date and gc_delay"""
|
|
self.branch.config_id = self.env.ref('runbot.runbot_build_config_default')
|
|
build = self.create_build({
|
|
'branch_id': self.branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'local_state': 'done'
|
|
})
|
|
|
|
child_build = self.create_build({
|
|
'branch_id': self.branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'extra_params': '2',
|
|
'parent_id': build.id,
|
|
'local_state': 'done'
|
|
})
|
|
|
|
# verify that the gc_day is set 30 days later (29 days since we should be a few microseconds later)
|
|
delta = fields.Datetime.from_string(build.gc_date) - datetime.datetime.now()
|
|
self.assertEqual(delta.days, 29)
|
|
child_delta = fields.Datetime.from_string(child_build.gc_date) - datetime.datetime.now()
|
|
self.assertEqual(child_delta.days, 14)
|
|
|
|
# Keep child build ten days more
|
|
child_build.gc_delay = 10
|
|
child_delta = fields.Datetime.from_string(child_build.gc_date) - datetime.datetime.now()
|
|
self.assertEqual(child_delta.days, 24)
|
|
|
|
# test the real _local_cleanup method
|
|
self.stop_patcher('_local_cleanup_patcher')
|
|
self.start_patcher('build_local_pgadmin_cursor_patcher', 'odoo.addons.runbot.models.build.local_pgadmin_cursor')
|
|
self.start_patcher('build_os_listdirr_patcher', 'odoo.addons.runbot.models.build.os.listdir')
|
|
dbname = '%s-foobar' % build.dest
|
|
self.start_patcher('list_local_dbs_patcher', 'odoo.addons.runbot.models.build.list_local_dbs', return_value=[dbname])
|
|
|
|
build._local_cleanup()
|
|
self.assertFalse(self.patchers['_local_pg_dropdb_patcher'].called)
|
|
build.job_end = datetime.datetime.now() - datetime.timedelta(days=31)
|
|
build._local_cleanup()
|
|
self.patchers['_local_pg_dropdb_patcher'].assert_called_with(dbname)
|
|
|
|
def test_repo_gc_testing(self):
|
|
""" test that builds are killed when room is needed on a host """
|
|
host = self.env['runbot.host'].create({
|
|
'name': 'runbot_xxx',
|
|
'nb_worker': 2
|
|
})
|
|
|
|
build_other_host = self.create_build({
|
|
'branch_id': self.branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'local_state': 'testing',
|
|
'host': 'runbot_yyy'
|
|
})
|
|
|
|
child_build = self.create_build({
|
|
'branch_id': self.branch.id,
|
|
'name': 'aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa',
|
|
'local_state': 'testing',
|
|
'host': 'runbot_xxx',
|
|
'extra_params': '2',
|
|
'parent_id': build_other_host.id
|
|
})
|
|
|
|
self.branch.repo_id._gc_testing(host)
|
|
self.assertFalse(build_other_host.requested_action)
|
|
self.assertFalse(child_build.requested_action)
|
|
|
|
build_same_branch = self.create_build({
|
|
'branch_id': self.branch_11.id,
|
|
'name': 'bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb',
|
|
'local_state': 'testing',
|
|
'host': 'runbot_xxx',
|
|
})
|
|
|
|
self.branch.repo_id._gc_testing(host)
|
|
self.assertFalse(build_same_branch.requested_action)
|
|
|
|
build_pending = self.create_build({
|
|
'branch_id': self.branch.id,
|
|
'name': 'deadbeafffffffffffffffffffffffffffffffff',
|
|
'local_state': 'pending',
|
|
})
|
|
|
|
self.branch.repo_id._gc_testing(host)
|
|
self.assertEqual(build_other_host.requested_action, 'deathrow')
|
|
self.assertEqual(child_build.requested_action, 'deathrow')
|
|
self.assertFalse(build_same_branch.requested_action)
|
|
self.assertFalse(build_pending.requested_action)
|
|
|
|
@patch('odoo.addons.runbot.models.build._logger')
|
|
def test_build_skip(self, mock_logger):
|
|
"""test build is skipped"""
|
|
build = self.create_build({
|
|
'branch_id': self.branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'port': '1234',
|
|
})
|
|
build._skip()
|
|
self.assertEqual(build.local_state, 'done')
|
|
self.assertEqual(build.local_result, 'skipped')
|
|
|
|
other_build = self.create_build({
|
|
'branch_id': self.branch.id,
|
|
'name': 'deadbeef0000ffffffffffffffffffffffffffff',
|
|
'port': '1234',
|
|
})
|
|
other_build._skip(reason='A good reason')
|
|
self.assertEqual(other_build.local_state, 'done')
|
|
self.assertEqual(other_build.local_result, 'skipped')
|
|
log_first_part = '%s skip %%s' % (other_build.dest)
|
|
mock_logger.debug.assert_called_with(log_first_part, 'A good reason')
|
|
|
|
def test_ask_kill_duplicate(self):
|
|
""" Test that the _ask_kill method works on duplicate when they are related PR/branch"""
|
|
|
|
branch = self.Branch.create({
|
|
'repo_id': self.repo.id,
|
|
'name': 'refs/heads/master-test-branch-xxx'
|
|
})
|
|
|
|
pr = self.Branch.create({
|
|
'repo_id': self.repo.id,
|
|
'name': 'refs/pull/1234',
|
|
'pull_head_name': 'odoo:master-test-branch-xxx'
|
|
})
|
|
|
|
build1 = self.create_build({
|
|
'branch_id': branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
})
|
|
build2 = self.create_build({
|
|
'branch_id': pr.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
})
|
|
build2.write({'local_state': 'duplicate', 'duplicate_id': build1.id}) # this may not be usefull if we detect duplicate in same repo.
|
|
|
|
self.assertEqual(build1.local_state, 'pending')
|
|
build2._ask_kill()
|
|
self.assertEqual(build1.local_state, 'done', 'A killed pending duplicate build should mark the real build as done')
|
|
self.assertEqual(build1.local_result, 'skipped', 'A killed pending duplicate build should mark the real build as skipped')
|
|
|
|
def test_children(self):
|
|
build1 = self.create_build({
|
|
'branch_id': self.branch_10.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
})
|
|
build1_1 = self.create_build({
|
|
'branch_id': self.branch_10.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'parent_id': build1.id,
|
|
'hidden': True,
|
|
'extra_params': '2', # avoid duplicate
|
|
})
|
|
build1_2 = self.create_build({
|
|
'branch_id': self.branch_10.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'parent_id': build1.id,
|
|
'extra_params': '3',
|
|
})
|
|
build1_1_1 = self.create_build({
|
|
'branch_id': self.branch_10.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'parent_id': build1_1.id,
|
|
'extra_params': '4',
|
|
})
|
|
build1_1_2 = self.create_build({
|
|
'branch_id': self.branch_10.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'parent_id': build1_1.id,
|
|
'extra_params': '5',
|
|
})
|
|
|
|
def assert_state(global_state, build):
|
|
self.assertEqual(build.global_state, global_state)
|
|
|
|
assert_state('pending', build1)
|
|
assert_state('pending', build1_1)
|
|
assert_state('pending', build1_2)
|
|
assert_state('pending', build1_1_1)
|
|
assert_state('pending', build1_1_2)
|
|
|
|
build1.local_state = 'testing'
|
|
build1_1.local_state = 'testing'
|
|
build1.local_state = 'done'
|
|
build1_1.local_state = 'done'
|
|
|
|
assert_state('waiting', build1)
|
|
assert_state('waiting', build1_1)
|
|
assert_state('pending', build1_2)
|
|
assert_state('pending', build1_1_1)
|
|
assert_state('pending', build1_1_2)
|
|
|
|
build1_1_1.local_state = 'testing'
|
|
|
|
assert_state('waiting', build1)
|
|
assert_state('waiting', build1_1)
|
|
assert_state('pending', build1_2)
|
|
assert_state('testing', build1_1_1)
|
|
assert_state('pending', build1_1_2)
|
|
|
|
build1_2.local_state = 'testing'
|
|
|
|
assert_state('waiting', build1)
|
|
assert_state('waiting', build1_1)
|
|
assert_state('testing', build1_2)
|
|
assert_state('testing', build1_1_1)
|
|
assert_state('pending', build1_1_2)
|
|
|
|
build1_2.local_state = 'testing' # writing same state a second time
|
|
|
|
assert_state('waiting', build1)
|
|
assert_state('waiting', build1_1)
|
|
assert_state('testing', build1_2)
|
|
assert_state('testing', build1_1_1)
|
|
assert_state('pending', build1_1_2)
|
|
|
|
build1_1_2.local_state = 'done'
|
|
build1_1_1.local_state = 'done'
|
|
build1_2.local_state = 'done'
|
|
|
|
assert_state('done', build1)
|
|
assert_state('done', build1_1)
|
|
assert_state('done', build1_2)
|
|
assert_state('done', build1_1_1)
|
|
assert_state('done', build1_1_2)
|
|
|
|
def test_duplicate_childrens(self):
|
|
build_old = self.create_build({
|
|
'branch_id': self.branch_10.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'extra_params': '0',
|
|
'local_state': 'done'
|
|
})
|
|
build_parent = self.create_build({
|
|
'branch_id': self.branch_10.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'extra_params': '1',
|
|
})
|
|
build_child = self.create_build({
|
|
'branch_id': self.branch_10.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
'parent_id': build_parent.id,
|
|
'extra_params': '0',
|
|
})
|
|
build_parent.local_state = 'done'
|
|
self.assertEqual(build_child.local_state, 'duplicate')
|
|
self.assertEqual(build_child.duplicate_id, build_old)
|
|
self.assertEqual(build_child.global_state, 'done')
|
|
self.assertEqual(build_parent.global_state, 'done')
|
|
|
|
|
|
class TestClosestBranch(RunbotCase):
|
|
|
|
def branch_description(self, branch):
|
|
branch_type = 'pull' if 'pull' in branch.name else 'branch'
|
|
return '%s %s:%s' % (branch_type, branch.repo_id.name.split(':')[-1], branch.name.split('/')[-1])
|
|
|
|
def assertClosest(self, branch, closest):
|
|
extra_repo = branch.repo_id.dependency_ids[0]
|
|
self.assertEqual(closest, branch._get_closest_branch(extra_repo.id), "build on %s didn't had the extected closest branch" % self.branch_description(branch))
|
|
|
|
def assertDuplicate(self, branch1, branch2, b1_closest=None, b2_closest=None, noDuplicate=False):
|
|
"""
|
|
Test that the creation of a build on branch1 and branch2 detects duplicate, no matter the order.
|
|
Also test that build on branch1 closest_branch_name result is b1_closest if given
|
|
Also test that build on branch2 closest_branch_name result is b2_closest if given
|
|
"""
|
|
closest = {
|
|
branch1: b1_closest,
|
|
branch2: b2_closest,
|
|
}
|
|
for b1, b2 in [(branch1, branch2), (branch2, branch1)]:
|
|
hash = '%s%s' % (b1.name, b2.name)
|
|
build1 = self.create_build({
|
|
'branch_id': b1.id,
|
|
'name': hash,
|
|
})
|
|
|
|
if b1_closest:
|
|
self.assertClosest(b1, closest[b1])
|
|
|
|
build2 = self.create_build({
|
|
'branch_id': b2.id,
|
|
'name': hash,
|
|
})
|
|
|
|
if b2_closest:
|
|
self.assertClosest(b2, closest[b2])
|
|
if noDuplicate:
|
|
self.assertNotEqual(build2.local_state, 'duplicate')
|
|
self.assertFalse(build2.duplicate_id, "build on %s was detected as duplicate of build %s" % (self.branch_description(b2), build2.duplicate_id))
|
|
else:
|
|
self.assertEqual(build2.duplicate_id.id, build1.id, "build on %s wasn't detected as duplicate of build on %s" % (self.branch_description(b2), self.branch_description(b1)))
|
|
self.assertEqual(build2.local_state, 'duplicate')
|
|
|
|
def assertNoDuplicate(self, branch1, branch2, b1_closest=None, b2_closest=None):
|
|
self.assertDuplicate(branch1, branch2, b1_closest=b1_closest, b2_closest=b2_closest, noDuplicate=True)
|
|
|
|
def setUp(self):
|
|
""" Setup repositories that mimick the Odoo repos """
|
|
super(TestClosestBranch, self).setUp()
|
|
self.Repo = self.env['runbot.repo']
|
|
self.community_repo = self.Repo.create({'name': 'bla@example.com:odoo/odoo', 'token': '1'})
|
|
self.enterprise_repo = self.Repo.create({'name': 'bla@example.com:odoo/enterprise', 'token': '1'})
|
|
self.community_dev_repo = self.Repo.create({'name': 'bla@example.com:odoo-dev/odoo', 'token': '1'})
|
|
self.enterprise_dev_repo = self.Repo.create({'name': 'bla@example.com:odoo-dev/enterprise', 'token': '1'})
|
|
|
|
# tweak duplicates links between repos
|
|
self.community_repo.duplicate_id = self.community_dev_repo.id
|
|
self.community_dev_repo.duplicate_id = self.community_repo.id
|
|
self.enterprise_repo.duplicate_id = self.enterprise_dev_repo.id
|
|
self.enterprise_dev_repo.duplicate_id = self.enterprise_repo.id
|
|
|
|
# create depenedencies to find Odoo server
|
|
self.enterprise_repo.dependency_ids = self.community_repo
|
|
self.enterprise_dev_repo.dependency_ids = self.community_dev_repo
|
|
|
|
# Create some sticky branches
|
|
self.Branch = self.env['runbot.branch']
|
|
self.branch_odoo_master = self.Branch.create({
|
|
'repo_id': self.community_repo.id,
|
|
'name': 'refs/heads/master',
|
|
'sticky': True,
|
|
})
|
|
self.branch_odoo_10 = self.Branch.create({
|
|
'repo_id': self.community_repo.id,
|
|
'name': 'refs/heads/10.0',
|
|
'sticky': True,
|
|
})
|
|
self.branch_odoo_11 = self.Branch.create({
|
|
'repo_id': self.community_repo.id,
|
|
'name': 'refs/heads/11.0',
|
|
'sticky': True,
|
|
})
|
|
|
|
self.branch_enterprise_master = self.Branch.create({
|
|
'repo_id': self.enterprise_repo.id,
|
|
'name': 'refs/heads/master',
|
|
'sticky': True,
|
|
})
|
|
self.branch_enterprise_10 = self.Branch.create({
|
|
'repo_id': self.enterprise_repo.id,
|
|
'name': 'refs/heads/10.0',
|
|
'sticky': True,
|
|
})
|
|
self.branch_enterprise_11 = self.Branch.create({
|
|
'repo_id': self.enterprise_repo.id,
|
|
'name': 'refs/heads/11.0',
|
|
'sticky': True,
|
|
})
|
|
|
|
self.Build = self.env['runbot.build']
|
|
|
|
def test_pr_is_duplicate(self):
|
|
""" test PR is a duplicate of a dev branch build """
|
|
|
|
mock_github = self.patchers['github_patcher']
|
|
mock_github.return_value = {
|
|
'head': {'label': 'odoo-dev:10.0-fix-thing-moc'},
|
|
'base': {'ref': '10.0'},
|
|
'state': 'open'
|
|
}
|
|
|
|
dev_branch = self.Branch.create({
|
|
'repo_id': self.community_dev_repo.id,
|
|
'name': 'refs/heads/10.0-fix-thing-moc'
|
|
})
|
|
pr = self.Branch.create({
|
|
'repo_id': self.community_repo.id,
|
|
'name': 'refs/pull/12345'
|
|
})
|
|
self.assertDuplicate(dev_branch, pr)
|
|
|
|
def test_closest_branch_01(self):
|
|
""" test find a matching branch in a target repo based on branch name """
|
|
|
|
self.Branch.create({
|
|
'repo_id': self.community_dev_repo.id,
|
|
'name': 'refs/heads/10.0-fix-thing-moc'
|
|
})
|
|
addons_branch = self.Branch.create({
|
|
'repo_id': self.enterprise_dev_repo.id,
|
|
'name': 'refs/heads/10.0-fix-thing-moc'
|
|
})
|
|
|
|
self.assertEqual((addons_branch, 'exact'), addons_branch._get_closest_branch(self.enterprise_dev_repo.id))
|
|
|
|
def test_closest_branch_02(self):
|
|
""" test find two matching PR having the same head name """
|
|
|
|
mock_github = self.patchers['github_patcher']
|
|
mock_github.return_value = {
|
|
# "head label" is the repo:branch where the PR comes from
|
|
# "base ref" is the target of the PR
|
|
'head': {'label': 'odoo-dev:bar_branch'},
|
|
'base': {'ref': 'saas-12.2'},
|
|
'state': 'open'
|
|
}
|
|
|
|
# update to avoid test to break. we asume that bar_branch exists.
|
|
# we may want to modify the branch creation to ensure that
|
|
# -> first make all branches
|
|
# -> then make all builds
|
|
community_branch = self.Branch.create({
|
|
'repo_id': self.community_dev_repo.id,
|
|
'name': 'refs/heads/bar_branch'
|
|
})
|
|
|
|
# Create PR in community
|
|
community_pr = self.Branch.create({
|
|
'repo_id': self.community_repo.id,
|
|
'name': 'refs/pull/123456'
|
|
})
|
|
enterprise_pr = self.Branch.create({
|
|
'repo_id': self.enterprise_repo.id,
|
|
'name': 'refs/pull/789101'
|
|
})
|
|
self.assertEqual((community_branch, 'exact PR'), enterprise_pr._get_closest_branch(self.community_repo.id))
|
|
|
|
def test_closest_branch_02_improved(self):
|
|
""" test that a PR in enterprise with a matching PR in Community
|
|
uses the matching one"""
|
|
|
|
mock_github = self.patchers['github_patcher']
|
|
|
|
com_dev_branch = self.Branch.create({
|
|
'repo_id': self.community_dev_repo.id,
|
|
'name': 'refs/heads/saas-12.2-blabla'
|
|
})
|
|
|
|
ent_dev_branch = self.Branch.create({
|
|
'repo_id': self.enterprise_dev_repo.id,
|
|
'name': 'refs/heads/saas-12.2-blabla'
|
|
})
|
|
|
|
def github_side_effect(url, **kwargs):
|
|
# "head label" is the repo:branch where the PR comes from
|
|
# "base ref" is the target of the PR
|
|
if url.endswith('/pulls/3721'):
|
|
return {
|
|
'head': {'label': 'odoo-dev:saas-12.2-blabla'},
|
|
'base': {'ref': 'saas-12.2'},
|
|
'state': 'open'
|
|
}
|
|
elif url.endswith('/pulls/32156'):
|
|
return {
|
|
'head': {'label': 'odoo-dev:saas-12.2-blabla'},
|
|
'base': {'ref': 'saas-12.2'},
|
|
'state': 'open'
|
|
}
|
|
else:
|
|
self.assertTrue(False)
|
|
|
|
mock_github.side_effect = github_side_effect
|
|
|
|
ent_pr = self.Branch.create({
|
|
'repo_id': self.enterprise_repo.id,
|
|
'name': 'refs/pull/3721'
|
|
})
|
|
|
|
self.Branch.create({
|
|
'repo_id': self.community_repo.id,
|
|
'name': 'refs/pull/32156'
|
|
})
|
|
with patch('odoo.addons.runbot.models.repo.runbot_repo._git_rev_parse', new=rev_parse):
|
|
self.assertDuplicate(
|
|
ent_dev_branch,
|
|
ent_pr,
|
|
(com_dev_branch, 'exact'),
|
|
(com_dev_branch, 'exact PR')
|
|
)
|
|
|
|
def test_closest_branch_03(self):
|
|
""" test find a branch based on dashed prefix"""
|
|
addons_branch = self.Branch.create({
|
|
'repo_id': self.enterprise_dev_repo.id,
|
|
'name': 'refs/heads/10.0-fix-blah-blah-moc'
|
|
})
|
|
self.assertEqual((self.branch_odoo_10, 'prefix'), addons_branch._get_closest_branch(self.community_repo.id))
|
|
|
|
def test_closest_branch_03_05(self):
|
|
""" test that a PR in enterprise without a matching PR in Community
|
|
and no branch in community"""
|
|
mock_github = self.patchers['github_patcher']
|
|
# comm_repo = self.repo
|
|
# self.repo.write({'token': 1})
|
|
|
|
ent_dev_branch = self.Branch.create({
|
|
'repo_id': self.enterprise_dev_repo.id,
|
|
'name': 'refs/heads/saas-12.2-blabla'
|
|
})
|
|
|
|
def github_side_effect(url, **kwargs):
|
|
if url.endswith('/pulls/3721'):
|
|
return {
|
|
'head': {'label': 'odoo-dev:saas-12.2-blabla'},
|
|
'base': {'ref': 'saas-12.2'},
|
|
'state': 'open'
|
|
}
|
|
elif url.endswith('/pulls/32156'):
|
|
return {
|
|
'head': {'label': 'odoo-dev:saas-12.2-blabla'},
|
|
'base': {'ref': 'saas-12.2'},
|
|
'state': 'open'
|
|
}
|
|
else:
|
|
self.assertTrue(False)
|
|
|
|
mock_github.side_effect = github_side_effect
|
|
|
|
com_branch = self.Branch.create({
|
|
'repo_id': self.community_repo.id,
|
|
'name': 'refs/heads/saas-12.2'
|
|
})
|
|
|
|
ent_pr = self.Branch.create({
|
|
'repo_id': self.enterprise_repo.id,
|
|
'name': 'refs/pull/3721'
|
|
})
|
|
with patch('odoo.addons.runbot.models.repo.runbot_repo._git_rev_parse', new=rev_parse):
|
|
self.assertDuplicate(
|
|
ent_pr,
|
|
ent_dev_branch,
|
|
(com_branch, 'pr_target'),
|
|
(com_branch, 'prefix'),
|
|
)
|
|
|
|
def test_closest_branch_04(self):
|
|
""" test that a PR in enterprise without a matching PR in Community
|
|
uses the corresponding exact branch in community"""
|
|
mock_github = self.patchers['github_patcher']
|
|
|
|
com_dev_branch = self.Branch.create({
|
|
'repo_id': self.community_dev_repo.id,
|
|
'name': 'refs/heads/saas-12.2-blabla'
|
|
})
|
|
|
|
ent_dev_branch = self.Branch.create({
|
|
'repo_id': self.enterprise_dev_repo.id,
|
|
'name': 'refs/heads/saas-12.2-blabla'
|
|
})
|
|
|
|
def github_side_effect(*args, **kwargs):
|
|
return {
|
|
'head': {'label': 'odoo-dev:saas-12.2-blabla'},
|
|
'base': {'ref': 'saas-12.2'},
|
|
'state': 'open'
|
|
}
|
|
|
|
mock_github.side_effect = github_side_effect
|
|
|
|
ent_pr = self.Branch.create({
|
|
'repo_id': self.enterprise_repo.id,
|
|
'name': 'refs/pull/3721'
|
|
})
|
|
with patch('odoo.addons.runbot.models.repo.runbot_repo._git_rev_parse', new=rev_parse):
|
|
self.assertDuplicate(
|
|
ent_dev_branch,
|
|
ent_pr,
|
|
(com_dev_branch, 'exact'),
|
|
(com_dev_branch, 'no PR')
|
|
)
|
|
|
|
def test_closest_branch_05(self):
|
|
""" test last resort value """
|
|
mock_github = self.patchers['github_patcher']
|
|
mock_github.return_value = {
|
|
'head': {'label': 'foo-dev:bar_branch'},
|
|
'base': {'ref': '10.0'},
|
|
'state': 'open'
|
|
}
|
|
|
|
server_pr = self.Branch.create({
|
|
'repo_id': self.community_repo.id,
|
|
'name': 'refs/pull/123456'
|
|
})
|
|
|
|
# trigger compute and ensure that mock_github is used. (using correct side effect would work too)
|
|
self.assertEqual(server_pr.pull_head_name, 'foo-dev:bar_branch')
|
|
|
|
mock_github.return_value = {
|
|
'head': {'label': 'foo-dev:foobar_branch'},
|
|
'base': {'ref': '10.0'},
|
|
'state': 'open'
|
|
}
|
|
addons_pr = self.Branch.create({
|
|
'repo_id': self.enterprise_repo.id,
|
|
'name': 'refs/pull/789101'
|
|
})
|
|
self.assertEqual(addons_pr.pull_head_name, 'foo-dev:foobar_branch')
|
|
closest = addons_pr._get_closest_branch(self.community_repo.id)
|
|
self.assertEqual((self.branch_odoo_10, 'pr_target'), addons_pr._get_closest_branch(self.community_repo.id))
|
|
|
|
def test_closest_branch_05_master(self):
|
|
""" test last resort value when nothing common can be found"""
|
|
|
|
addons_branch = self.Branch.create({
|
|
'repo_id': self.enterprise_dev_repo.id,
|
|
'name': 'refs/head/badref-fix-foo'
|
|
})
|
|
self.assertEqual((self.branch_odoo_master, 'default'), addons_branch._get_closest_branch(self.community_repo.id))
|
|
|
|
def test_no_duplicate_update(self):
|
|
"""push a dev branch in enterprise with same head as sticky, but with a matching branch in community"""
|
|
community_sticky_branch = self.Branch.create({
|
|
'repo_id': self.community_repo.id,
|
|
'name': 'refs/heads/saas-12.2',
|
|
'sticky': True,
|
|
})
|
|
community_dev_branch = self.Branch.create({
|
|
'repo_id': self.community_dev_repo.id,
|
|
'name': 'refs/heads/saas-12.2-dev1',
|
|
})
|
|
enterprise_sticky_branch = self.Branch.create({
|
|
'repo_id': self.enterprise_repo.id,
|
|
'name': 'refs/heads/saas-12.2',
|
|
'sticky': True,
|
|
})
|
|
enterprise_dev_branch = self.Branch.create({
|
|
'repo_id': self.enterprise_dev_repo.id,
|
|
'name': 'refs/heads/saas-12.2-dev1'
|
|
})
|
|
# we shouldn't have duplicate since community_dev_branch exists
|
|
with patch('odoo.addons.runbot.models.repo.runbot_repo._git_rev_parse', new=rev_parse):
|
|
# lets create an old enterprise build
|
|
self.create_build({
|
|
'branch_id': enterprise_sticky_branch.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
})
|
|
self.assertNoDuplicate(
|
|
enterprise_sticky_branch,
|
|
enterprise_dev_branch,
|
|
(community_sticky_branch, 'exact'),
|
|
(community_dev_branch, 'exact'),
|
|
)
|
|
|
|
def test_external_pr_closest_branch(self):
|
|
""" test last resort value target_name"""
|
|
mock_github = self.patchers['github_patcher']
|
|
mock_github.return_value = {
|
|
'head': {'label': 'external_repo:11.0-fix'},
|
|
'base': {'ref': '11.0'},
|
|
'state': 'open'
|
|
}
|
|
enterprise_pr = self.Branch.create({
|
|
'repo_id': self.enterprise_repo.id,
|
|
'name': 'refs/pull/123456'
|
|
})
|
|
dependency_repo = self.enterprise_repo.dependency_ids[0]
|
|
closest_branch = enterprise_pr._get_closest_branch(dependency_repo.id)
|
|
self.assertEqual(enterprise_pr._get_closest_branch(dependency_repo.id), (self.branch_odoo_11, 'pr_target'))
|
|
|
|
def test_external_pr_with_comunity_pr_closest_branch(self):
|
|
""" test matching external pr """
|
|
mock_github = self.patchers['github_patcher']
|
|
mock_github.return_value = {
|
|
'head': {'label': 'external_dev_repo:11.0-fix'},
|
|
'base': {'ref': '11.0'},
|
|
'state': 'open'
|
|
}
|
|
community_pr = self.Branch.create({
|
|
'repo_id': self.community_repo.id,
|
|
'name': 'refs/pull/123456'
|
|
})
|
|
mock_github.return_value = {
|
|
'head': {'label': 'external_dev_repo:11.0-fix'}, # if repo doenst match, it wont work, maybe a fix to do here?
|
|
'base': {'ref': '11.0'},
|
|
'state': 'open'
|
|
}
|
|
enterprise_pr = self.Branch.create({
|
|
'repo_id': self.enterprise_repo.id,
|
|
'name': 'refs/pull/123'
|
|
})
|
|
with patch('odoo.addons.runbot.models.repo.runbot_repo._git_rev_parse', new=rev_parse):
|
|
build = self.create_build({
|
|
'branch_id': enterprise_pr.id,
|
|
'name': 'd0d0caca0000ffffffffffffffffffffffffffff',
|
|
})
|
|
dependency_repo = build.repo_id.dependency_ids[0]
|
|
self.assertEqual(build.branch_id._get_closest_branch(dependency_repo.id), (community_pr, 'exact PR'))
|
|
# this is working here because pull_head_name is set, but on runbot pull_head_name is empty for external pr. why?
|