[IMP] runbot: add a codeowner model

As a custom codeowner system was successfully implemented in a python
step on our runbot instance, it's now time to have a real model for
that.

This commit adds a skeleton Codeowner model in order to be used for a
basic usage.

This should be improved in the future after some battle testing.
This commit is contained in:
Christophe Monniez 2021-11-17 22:28:14 +01:00 committed by xdo
parent 89dcb52215
commit 91efbec943
6 changed files with 121 additions and 0 deletions

View File

@ -41,6 +41,7 @@
'views/build_error_views.xml',
'views/build_views.xml',
'views/bundle_views.xml',
'views/codeowner_views.xml',
'views/commit_views.xml',
'views/config_views.xml',
'views/dashboard_views.xml',

View File

@ -6,6 +6,7 @@ from . import build
from . import build_config
from . import build_error
from . import bundle
from . import codeowner
from . import commit
from . import database
from . import dockerfile
@ -22,5 +23,6 @@ from . import upgrade
from . import user
from . import version
# those imports have to be at the end otherwise the sql view cannot be initialised
from . import build_stat
from . import build_stat_regex

View File

@ -0,0 +1,30 @@
import ast
import re
from odoo import models, fields, api
from odoo.exceptions import ValidationError
class Codeowner(models.Model):
_name = 'runbot.codeowner'
_description = "Notify github teams based on filenames regex"
_inherit = "mail.thread"
project_id = fields.Many2one('runbot.project', required=True)
regex = fields.Char('Regular Expression', help='Regex to match full file paths', required=True, tracking=True)
github_teams = fields.Char(help='Comma separated list of github teams to notify', required=True, tracking=True)
team_id = fields.Many2one('runbot.team', help='Not mandatory runbot team')
version_domain = fields.Char('Version Domain', help='Codeowner only applies to the filtered versions')
@api.constrains('regex')
def _validate_regex(self):
for rec in self:
try:
r = re.compile(rec.regex)
except re.error as e:
raise ValidationError("Unable to compile regular expression: %s" % e)
def _get_version_domain(self):
""" Helper to get the evaluated version domain """
self.ensure_one()
return ast.eval(self.version_domain) if self.version_domain else []

View File

@ -111,3 +111,6 @@ access_runbot_upgrade_exception_admin,access_runbot_upgrade_exception_admin,runb
access_runbot_dockerfile_user,access_runbot_dockerfile_user,runbot.model_runbot_dockerfile,runbot.group_user,1,0,0,0
access_runbot_dockerfile_admin,access_runbot_dockerfile_admin,runbot.model_runbot_dockerfile,runbot.group_runbot_admin,1,1,1,1
access_runbot_codeowner_admin,runbot_codeowner_admin,runbot.model_runbot_codeowner,runbot.group_runbot_admin,1,1,1,1
access_runbot_codeowner_user,runbot_codeowner_user,runbot.model_runbot_codeowner,group_user,1,0,0,0

1 id name model_id:id group_id:id perm_read perm_write perm_create perm_unlink
111
112
113
114
115
116

View File

@ -191,3 +191,35 @@ class TestBuildError(RunbotCase):
})
self.assertEqual(dashboard.build_ids, failed_build)
class TestCodeOwner(RunbotCase):
def setUp(self):
super().setUp()
self.cow_deb = self.env['runbot.codeowner'].create({
'project_id' : self.project.id,
'github_teams': 'runbot',
'regex': '.*debian.*'
})
self.cow_web = self.env['runbot.codeowner'].create({
'project_id' : self.project.id,
'github_teams': 'website',
'regex': '.*website.*'
})
self.cow_crm = self.env['runbot.codeowner'].create({
'project_id' : self.project.id,
'github_teams': 'crm',
'regex': '.*crm.*'
})
self.cow_all = self.cow_deb | self.cow_web | self.cow_crm
def test_codeowner_invalid_regex(self):
with self.assertRaises(ValidationError):
self.env['runbot.codeowner'].create({
'project_id': self.project.id,
'regex': '*debian.*',
'github_teams': 'rd-test'
})

View File

@ -0,0 +1,53 @@
<odoo>
<data>
<record id="codeowner_form" model="ir.ui.view">
<field name="name">runbot.codeowner.form</field>
<field name="model">runbot.codeowner</field>
<field name="arch" type="xml">
<form string="Host">
<sheet>
<group>
<field name="project_id"/>
<field name="team_id"/>
<field name="github_teams"/>
<field name="regex"/>
<field name="version_domain" widget="domain" options="{'model': 'runbot.version', 'in_dialog': True, 'operators': ['in','=', '&lt;', '&gt;']}"/>
</group>
</sheet>
<div class="oe_chatter">
<field name="message_follower_ids" widget="mail_followers"/>
<field name="message_ids" widget="mail_thread"/>
</div>
</form>
</field>
</record>
<record id="view_codeowner_tree" model="ir.ui.view">
<field name="name">runbot.codeowner.tree</field>
<field name="model">runbot.codeowner</field>
<field name="arch" type="xml">
<tree string="Codeowners">
<field name="project_id"/>
<field name="team_id"/>
<field name="version_domain"/>
<field name="regex"/>
<field name="github_teams"/>
</tree>
</field>
</record>
<record id="open_view_codeowner_tree" model="ir.actions.act_window">
<field name="name">Codeowner</field>
<field name="res_model">runbot.codeowner</field>
<field name="view_mode">tree,form</field>
</record>
<menuitem
name="CodeOwners"
id="runbot_menu_codeowner_tree"
parent="runbot_menu_manage_errors"
sequence="90"
action="open_view_codeowner_tree"
/>
</data>
</odoo>